Geometryczne wyzwanie


23

Wszyscy kochają geometrię. Dlaczego więc nie spróbujemy go zakodować w golfa? Wyzwanie to polega na przyjmowaniu liter i cyfr oraz tworzeniu kształtów w zależności od niego.

Wejście

Dane wejściowe będą miały postać (shapeIdentifier)(size)(inverter).

Ale czym są identyfikator kształtu, rozmiar i falownik?

Identyfikator kształtu jest identyfikatorem typu kształtu, który utworzysz za pomocą *s. Oto identyfikatory kształtów:

  • s - Plac
  • t - Trójkąt

Rozmiar będzie między 1-20, i jest to wielkość figury.

Falownik określa, czy kształt będzie odwrócony do góry nogami, co oznaczono znakiem a +lub a -. Uwaga: s3-== (równa się), s3+ponieważ kwadraty są symetryczne. Jednak t5-! = (Nie jest równy) t5+.

Końcowe białe znaki na wydruku są prawidłowe, ale początkowe białe znaki nie.

Przykłady wyników

Input: s3+
Output:
***
***
***

Input: t5+

Output:
  *
 ***
*****

Input: t3-
Output:
***
 *

Specjalne notatki

Trójkąt zawsze będzie miał nieparzystą liczbę, więc trójkąty zawsze kończą się 1 *na górze.

Rozmiar trójkąta jest wielkością podstawy, jeśli falownik jest, +i jest wielkością góry, jeśli falownik jest -.


3
Jako ktoś, kto teraz bierze Geometria (i przygotowuje się do finału Geometrii), mogę powiedzieć ze 100% pewnością: Geometria jest absolutnie, wcale nie zabawna ... D:
Ashwin Gupta

Odpowiedzi:


9

Pyth, 40 36 34 32 bajty

-1 bajt @isaacg

JstPz_W}\+zjl#m.[J*\*-J*}\tzyd;J

Średnik wewnątrz lambda jest teraz globalną wartością zmiennej lambda, cechą, która oszczędza jeden bajt.

                         Implicit: z = input
JstPz                    J = size.
_W }\+z                  Reverse if "+" in z
j l# m                J  Join the nonempty lines in map lambda d:... over range(J)
      .[J            ;   Pad the following with spaces (;) to length J
         *\*               "*", this many times:
            -J*}\tzyd        J if "t" not  in z,
                             otherwise the correct number for a triangle.

Wypróbuj tutaj .

Zestaw testowy .


1
O wiele za długo, ale pokonujesz Japt o 15 bajtów? Nie mogę się doczekać, aby zobaczyć, jak to będzie
grało w

Fajne rozwiązanie! Możesz zapisać bajt, zastępując qez\+go }\+z, ponieważ +może pojawić się tylko w ostatniej pozycji.
isaacg

6

Pyth, 38 bajtów

JsPtzj?}\szm*\*JJ_W}\-zm.[J*\*hyd;/hJ2

Zestaw testowy

Zasadniczo tak proste, jak to możliwe. Chciałbym móc połączyć logikę tych dwóch kształtów, ale obecnie jest to osobne.


5

JavaScript (ES6), 142 146 147

Edytuj 1 bajt zapisany thx @ ETHproditions Edytuj 2 bajty sve thx @ user81655

i=>([,a,b]=i.match`.(.+)(.)`,Array(l=i<'t'?+a:-~a/2).fill('*'.repeat(a)).map((r,i)=>l-a?(' '.repeat(i=b<'-'?--l:i)+r).slice(0,a-i):r).join`
`)

Test (uruchom w FireFox)

F=i=>(
  [,a,b]=i.match`.(.+)(.)`,
  Array(l=i<'t'?+a:-~a/2).fill('*'.repeat(a))
  .map((r,i)=>l-a?(' '.repeat(i=b<'-'?--l:i)+r).slice(0,a-i):r)
  .join`\n`
)

function test() { O.textContent=F(I.value) }

test()
Input: <input id=I oninput="test()" value="t11-"/>
<pre id=O></pre>


\d-> ., ponieważ gwarantowana jest dokładnie jedna cyfra przed i po
ETHproductions

@ETHproductions racja, dzięki
edc65

Miły. Myślę, że jest to optymalny algorytm w JS, nie mogę znaleźć krótszego.
ETHprodukcje

i.match(/.(.+)(.)/)->i.match`.(.+)(.)`
user81655

@ user81655 niezła podpowiedź, dzięki
edc65

5

Python 2, 106 bajtów

s=raw_input()
n=int(s[1:-1])
for i in[range(1,n+1,2),n*[n]][s<'t'][::2*('+'in s)-1]:print('*'*i).center(n)

Wyjście jest idealnym prostokątem, z każdą linią wypełnioną końcowymi spacjami, co, jak zakładam, jest w porządku na podstawie komentarzy w PO.

Uwaga: wciąż nie jestem pewien, czy inputw Pythonie 2 zezwala się na takie problemy ...


4

Japt, 62 60 55 52 51 bajtów

V=Us1 n;U<'t?Vo ç*pV):0oV2 £S²pY iY'*pV-X})·z2*!Uf-

Wypróbuj online!

Pierwszą rzeczą, którą musimy zrobić, to dowiedzieć się, jak duży powinien być nasz kształt. To jest dość proste:

      // Implicit: U = input string, S = space
V=    // Set variable V to
Us1   // everything after the first char of U,
n;    // converted to a number. This turns e.g. "12+" into 12.

Teraz organizujemy kształt wyniku:

U<'t?      // If U comes before "t" lexicographically (here, if the first char is "s"),
Vo         //  make a list of V items,
ç*pV)      //  and set each item to V asterisks.
:0oV2      // Otherwise, create the range [0, V) with steps of 2 (e.g. 7 -> [0,2,4,6]),
£       }) //  and map each item X and index Y to:
S²pY       //   Repeat 2 spaces Y times. This creates a string of Y*2 spaces.
iY'*pV-X   //   At position Y in this string (right in the middle), insert V-X asterisks.
·          // Join with newlines.

Do tej pory zadbaliśmy o wielkość i kształt produkcji. Pozostaje tylko obrót. Trójkąty są obecnie skierowane w górę, więc musimy je odwrócić, jeśli trzeci znak to +:

!Uf-    // Take the logical not of U.match("-").
        // If U contains "-", this returns false; otherwise, returns true.
2*      // Multiply by two. This converts true to 2, false to 0.
z       // Rotate the list 90° that many times.
        // Altogether, this turns the shape by 180° if necessary.

I przy dorozumianych wynikach, nasza praca tutaj jest zakończona. :-)


4

Python 2, 235 193 167 157 bajtów

Aktualizacja:

Dokonano znaczącej optymalizacji przy użyciu list list i str.center (). Mam wrażenie, że mogę coś jeszcze zrobić, a potem jeszcze raz na to spojrzę.

Aktualizacja 2

Zapisano 10 bajtów dzięki sugestiom Sherlocka9. Wielkie dzięki! :)

d=raw_input()
x=int(d[1:-1])
o="\n".join("*"*x for i in range(x))if d<"t"else"\n".join(("*"*i).center(x)for i in range(x,0,-2))
print o[::-1]if"+"in d else o

Stara odpowiedź

d=raw_input()
x=int(d[1:-1])
if "s" in d:
 for y in range(x):
    o+="*"*x+"\n"
 o=o[:-1]
else:
 b=0
 while x+1:
    o+=" "*b+"*"*x+" "*b+"\n"
    x-=2
    b+=1
 o=o[:-1]
 if d[-1]=="+":
    o=o[::-1]
print o

Całkiem proste podejście. Zapisywanie linii na linię w ciągu, który wypisuję na końcu. Trójkąty są zawsze rysowane odwrócone i w razie potrzeby są odwracane. Fakt, że możesz pomnożyć ciąg przez liczbę całkowitą, zaoszczędził mi wiele bajtów!

Później postaram się zagrać w golfa, w międzyczasie doceniłbym sugestie, ponieważ nie jestem jeszcze tak doświadczony.

edytuj: Grałem w golfa z pomocą komentarzy i kradzieży obliczeń rozmiaru z jednej z innych odpowiedzi na python. Myślę, że to wszystko, co mogę zrobić z tym algorytmem.


Jak liczyłeś Podczas korzystania wcdaje mi liczbę bajtów 235. Czy się mylę?
ბიმო

1
To rzeczywiście 235 bajtów. Porady dotyczące gry w golfa: używaj tabulatorów zamiast dwóch spacji, co jest poprawne w Pythonie 2 i powoduje wygaszenie 5 bajtów.
Klamka

Również nie musisz używać raw_input, użycie inputoszczędza 4 bajty. Ponadto nie potrzebujesz nawiasów w drugim wierszu, a to i nieużywanie zmiennej xw ogóle (używanie if"s"in d) pozwala zaoszczędzić kolejne 9 bajtów.
ბიმო

2
@DenkerAffe podczas liczenia w oknie, odejmij 1 bajt dla każdej nowej linii - nowe linie to 2 bajty w oknach, ale 1 bajt w innych środowiskach
edc65

1
Najpierw możesz usunąć []nawiasy klamrowe w każdym joinwywołaniu funkcji. Po drugie, if d<"t"elsejest krótszy i działa, ponieważ "s3+"<"t"<"t3+"w Pythonie. Po trzecie else"\n".joini .center(x)for. Brak miejsca. To nie jest wymagane. Po czwarte, print o[::-1]if"+"in d else ogdzie I łączyć rzeczy dla dwóch bajtów (jedna przestrzeń między ]i ifa drugi pomiędzy ifi "+".
Sherlock9

3

JavaScript, 220 bajtów.

q=s=>+s.slice(1,s.length-1);f=s=>s[0]=="s"?("*".repeat(q(s))+"\n").repeat(q(s)):Array.apply(0,Array(-~(q(s)/2))).map((_,n)=>(s[s.length-1]=="-"?~~(q(s)/2)-n:n)).map(n=>(" ".repeat(q(s)/2-n)+"*".repeat(n*2+1))).join("\n")

Biegnij z f(input here)

Wypróbuj tutaj!

Kwadraty mają końcowe znaki nowej linii, ale trójkąty nie. Wyjaśnienie:

q=s=>+s.slice(1,s.length-1);                                                                                                                                                                                                 Define a function, q, that takes returns the argument, without the first and last character, casted into an integer.
                            f=s=>                                                                                                                                                                                            Define a function, f, that takes one argument, s. (This is the main function)
                                 s[0]=="s"?                                                                                                                                                                                  If the first character of s is "s" then...
                                           ("*".repeat(q(s))     )                                                                                                                                                           Repeat the "*" character q(s) times.
                                           (                +"\n")                                                                                                                                                           Append a newline to that
                                                                  .repeat(q(s))                                                                                                                                              Repeat that q(s) times.
                                                                               :                                                                                                                                             Else... (the first character of s isn't "s")
                                                                                Array.apply(0,Array(          ))                                                                                                             Create an array of length...
                                                                                Array.apply(0,Array(-~(q(s)/2)))                                                                                                             floor(q(s)/2)+1
                                                                                                                .map((_,n)=>                                   )                                                             Map each element, _ with index n to...
                                                                                                                .map((_,n)=>(s[s.length-1]=="-"?              ))                                                             If the last element of s is "-" then...
                                                                                                                .map((_,n)=>(s[s.length-1]=="-"?~~(q(s)/2)-n  ))                                                             floor(q(s)/2)-n
                                                                                                                .map((_,n)=>(s[s.length-1]=="-"?            : ))                                                             Else...
                                                                                                                .map((_,n)=>(s[s.length-1]=="-"?             n))                                                             Just n
                                                                                                                                                                .map(n=>                                        )            Map each element into...
                                                                                                                                                                .map(n=>(" ".repeat(q(s)/2-n)                   )            Repeat " ", q(s)/2-n times.
                                                                                                                                                                .map(n=>(                   )+"*".repeat(n*2+1)))            Append "*", repeated 2n+1 times.
                                                                                                                                                                .map(n=>(" ".repeat(        )+"*".repeat(n*2+1))).join("\n") Join with newlines

Długość pierwszego wiersza wynosi 338 znaków. Wyświetlenie zajmuje mi półtora monitora.
isanae


1
Nie kliknę losowego linku tinyurl, ale sprawdź ponownie. W każdym razie staraj się unikać pasków przewijania w polach kodu, co znacznie utrudnia czytanie.
isanae

1
@Loovjo Myślę, że ma na myśli pierwszą linię wyjaśnienia. Zazwyczaj wciskam moje objaśnienie zamiast tego stylu dla odpowiedzi JavaScript, więc nie trzeba przewijać, aby zobaczyć połowę.
user81655

@ user81655 Tak, miałem na myśli wyjaśnienie. Teraz rozumiem zamieszanie!
isanae

3

Python 2, 157 132 bajtów

def f(s):
 S=int(s[1:-1])
 for n in([range(1,S+2,2),range(S,0,-2)]['-'in s],[S]*S)['s'in s]:
  print "{:^{S}}".format('*'*n,S=S)

Pierwsza próba potwierdziła, że +/-koniec był opcjonalny, pozbywanie się tego pozwoliło mi się ogolić

Chodzi tutaj o sporządzenie listy, która może zostać wrzucona do ogólnego wyniku. Najtrudniejszą częścią było oddzielenie długości od wejścia.


Aby uzyskać długość użyłem x=int(d[1]if len(d)<4 else d[1:3])d jako ciąg wejściowy. To o 5 bajtów mniej niż twoje rozwiązanie. Nadal jesteś daleko przed moim pythonowym pytaniem, musisz spróbować zrozumieć, co tam zrobiłeś i pokonać cię następnym razem! :)
Denker

1
W rzeczywistości x=int(d[1:-1])jest o wiele krótszy, po prostu widziałem to w innej odpowiedzi na python.
Denker

@DenkerAffe, z jakiegokolwiek powodu pamiętam, że falownik jest opcjonalny, więc to nie będzie działać, ale chyba właśnie to wymyśliłem
wnnmaw

2

Siatkówka , 102 85 bajtów

Liczba bajtów zakłada, że ​​kod źródłowy jest zakodowany jako ISO 8859-1.

\d+
$0$*:¶
^((\w)+):(:+)
$1$2$3$2¶$0
m`s$|:t

)`(.+)¶-(\D*)
-$2¶$1
m`^.

G`.
T`ts:` *

Wypróbuj online.

Spróbuję później zagrać w golfa.


Notepad ++ mówi, że kod jest 89 bajtów, a nie 85. Użyłem kodowania ISO-8859-1 i poszedł na Edycja> Konwersja> EOL formacie UNIX / Linux, używać \nzamiast \r\n. Base64 treści: XGQrCiQwJCo6wrYKXigoXHcpKyk6KDorKQokMSQyJDMkMsK2JDAKbWBzJHw6dAoKKWAoLispwrYtKFxEKikKLSQywrYkMQptYF4uCgpHYC4KVGB0czpgICo=(bezpośrednia kopia z Notepad ++). Co dziwne, każde rozwiązanie online daje mi 85 bajtów ... Hum ...
Ismael Miguel

@ IsmaelMiguel Musi być coś nie tak z tym, jak liczy się Notepad ++ . Są zdecydowanie jednobajtowe w ISO 8859-1 (o wartości 182).
Martin Ender

2

Poważnie, 54 bajty

,#i's=`≈;'**@½≈";#dXdXεj' +"£n`@`≈;'**n`@Iƒ('-=WXa0WXü

Wypróbuj online

,#i                                                    Take input, push chars separately
   's=                                   Iƒ            IF the first char is "s":
                                `      `@                run the quoted function
                                 ≈;'**n                  make list of n strings of n *'s
      `                       `@                       ELSE run the quoted function:
       ≈;                                                make two copies of int n
         '**                                             use one to make string of n *'s
            @½≈                                          cut the other in half (e.g. 5->2)
               "           "£n                           run n/2 times the quoted function:
                ;#                                        copy the string as list of chars
                  dXdX                                    discard the last 2 *'s
                      εj                                  join back into string
                        ' +                               prepend a space
                                           ('-=WX 0WX  IF the third character is "-":
                                                 a       invert the stack
                                                     ü pop and print the entire stack

@Mego: Widzisz to #dXdXεj? STRING SLICING ????


2

ES6, 178 172 159 bajtów

s=>(p=s.match(/d+|./g),u=n=+p[1],m=n+1>>1,t=' '.repeat(n)+'*'.repeat(n),v=s<'t'?0:p[2]<'-'?(u=m,1):-1,[...Array(s<'t'?n:m)].map(_=>t.substr(u,u,u+=v)).join`
`)

Działa to dzięki ciekawej obserwacji, którą poczyniłem. Jeśli powtórzysz nspacje i ngwiazdki, otrzymasz (np. Dla n=5):

     *****

Teraz weź podciągi o tym samym początku i długości:

     |*****| (5)
    | ***| (4)
   |  *| (3)

Te podciągi są dokładnie tymi, których potrzebujemy t5.

Edycja: Zapisano 6 bajtów dzięki @ edc65.

Edycja: Zapisano 13 bajtów dzięki ukryciu u+=vtrzeciego argumentu, substrdzięki czemu mogę uprościć inicjalizację.


@ThomasKwa Huh, po tym, jak stała się tkod obsługi okazało się, że wi ustał równoważne i że uratował mi na tyle bajtów, aby zabrać mnie z powrotem w dół do 178!
Neil

[,b,c]=s.matcha później s<'t'... powinien zapisać niektóre bajty (tylko Firefox)
edc65

@ edc65 Po prostu nie zapisanie dopasowania spozwala mi na użycie, s<'t'co pozwoliło mi zaoszczędzić 6 bajtów, dzięki.
Neil

2

MATL , 48 bajtów

' *'jt4Y2m)U1$l't'Gm?2MQ2/:1L3$)R!P!R'+'Gm?P]]Q)

Używa bieżącej wersji (10.1.0) języka / kompilatora.

Kod akceptuje znaki wejściowe w dowolnej kolejności: wszystkie s11+, 11s+a nawet 1+s1prawidłowe ciągi wejściowe.

EDYCJA (30 lipca 2016 r.): Połączony kod zastępuje się 1L3$)przez, Y)aby dostosować się do ostatnich zmian w języku

Wypróbuj online!

Wyjaśnienie

' *'        % push string. Will be indexed into to obtain final result
j           % input string
t           % duplicate
4Y2         % predefined literal string '0123456789'
m           % logical index of digits in input string
)           % index into input string to obtain substring with digits
U           % convert to number
1$l         % generate square of ones with that size
't'         % push character 't'
G           % push input string
m           % true if input string contains 't'
?           % if so...
  2M        % push argument of call to function `l`, i.e. square size
  Q2/       % add 1 and divide by 2. Call result T
  :         % generate vector [1, 2, ... T]
  1L        % predefined literal representing Matlab's `:` index
  3$)       % two dimensional index. Transforms square into rectangle
  R         % remove (set to zero) lower-left corner
  !P!       % flip horizontally
  R         % remove lower-left corner. This gives inverted triangle
  '+'       % push character '+'
  G         % push input
  m         % true if input contains '+'
  ?         % if so...
    P       % flip vertically
  ]         % end if
]           % end if
Q           % add 1. This gives array of values 1 and 2
)           % index string ' *' with this array to produce char array
            % implicitly display that char array

1

C, 259 bajtów

#define x(y);)putchar(y)
#define m(n)for(n=0;n++<
#define T {m(q)i x(32);m(q)s-i*2 x(42);puts("");}
main(q,v,i,s)char**v;{s=atoi(v[1]+1);if(*v[1]=='s')m(i)s*s x(42)&&!(i%s)&&puts("");else if(strchr(v[1],'+'))for(i=s/2+1;i-->0;)T else for(i=-1;i++<s/2+1;)T}

bez golfa

main(q,v,i,size)char**v; // neat way of declaring variables
{
    size=atoi(v[1]+1);
    if(*v[1]=='s')
    {
        for(i=0;i++<size*size;)
        {
            putchar(42); // returns 42 (true)
            if(!(i%size))
                puts("");
        }
    }
    else if(strchr(v[1],'+')) // if finds plus sign
    {
        for(i=size/2+1;i-->0;) // iterate the height of the triangle
        {
            for(q=0;q++<i;)putchar(32); // conveniently i is the number os spaces before each line
            for(q=0;q++<size-i*2;) putchar(42);
            puts("");
        }
    }
    else for(i=-1;i++<size/2+1;) // does the same as above but inverted order
    {
        for(q=0;q++<i;)putchar(32);
        for(q=0;q++<size-i*2;)putchar(42);
        puts("");
    }
}

Sugestie i krytyka są bardzo mile widziane.


1

Ruby, 99

->s{n=s[1,2].to_i
n.times{|i|d=(s.ord-115)*(s[-1]<=>?,)*(n-1-i*2)
d<1&&puts((?**(n+d)).center(n))}}

Oblicza kwadrat lub trójkąt o wysokości ni średniej szerokości n , sprawdzając nachylenie boków (więc obliczona szerokość trójkąta wynosi 2n-1 u podstawy, 1 na końcu). Ale drukuje tylko te wiersze, które nie przekraczają nznaków.

nie wziął udziału w programie testowym

f=->s{                         #take a string as an argument
  n=s[1,2].to_i                #take 2 characters starting at index 1 and convert to a number for the size
  n.times{|i|                  #iterate through n rows    
    d=                         #calculate how many stars "MORE THAN" n we need on a row
    (s.ord-115)*               #ascii code for 1st character of string - 115 : s-->0, t-->1
    (s[-1]<=>?,)*              #compare last character of input with comma character - --> +1 + --> -1
    (n-1-i*2)                  #row number * 2: 0 at centre, positive above it, negative below it
    d<1&&                      #only output if d is nonpositive (i.e we need less than n or exactly n stars)
    puts((?**(n+d)).center(n)) #print n+d stars, centred in a field of n characters padded by whitespace
  }
}

f[gets.chomp]

1

Jolf, 37 bajtów, niekonkurujący

Dodałem funkcje po opublikowaniu tego wyzwania, więc nie można tego uznać za akceptację. Jest to zakodowane w ISO-8859-7. Wypróbuj wszystkie przypadki testowe tutaj .

onFiΒ€ioSgiγ?='sn―sΒ'*―TΒ1'*?='-SZiγγ

Część 1: parsowanie łańcucha

onFiΒ€ioSgi
on          set n to
  Fi         the first entity of i (the shape identifier)
    Β       set Β (beta) to
     €i      the "inside" of i (in this case, the size) as a number
       oS   set S to
         gi  the last entity of i (the inverter)

Część 2: uzyskanie wyniku

γ?='sn―sΒ'*―TΒ1'*
γ                 set γ (gamma) to the result of the following expression
 ?='sn             if n is the character s,
      ―sΒ'*         then return a pattern "s" (a square) made with "*"s
           ―TΒ1'*    otherwise, return a pattern "T" (triangle) that is centered and
                     has a scale factor of 1, made with "*"s

Część 3: odwrócenie wyniku

?='-SZiγγ
?='-S     if S is a "-"
     Ziγ   return γ, inverted across its lines
        γ  otherwise, return γ untouched
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.