Disegna un esagono ASCII di lunghezza lato n
Dato un numero intero positivo \$n\$produce un esagono ASCII con lunghezza del lato diagonale \$n\$, come nella foto (nota che ci sono \$2n\$ _
caratteri sul bordo superiore)
Esempi:
\$n = 1\$
__
/ \
\__/
\$n = 2\$
____
/ \
/ \
\ /
\____/
\$n = 5\$
__________
/ \
/ \
/ \
/ \
/ \
\ /
\ /
\ /
\ /
\__________/
eccetera.
Il codice più breve in byte vince. Si applicano i normali metodi di input / output .
Risposte
Tela , 15 9 byte
_;1*⁸/∔╬
Provalo qui!
-6 byte dopo aver corretto il programma.
Disegna un quarto dell'esagono e quadricromia.
Python 2 , 92 byte
k=n=input()
while 1:a=k^k>>n;print" "*a+"\/_"[k/n-2]+"_ "[-n<k<n]*2*(2*n+~a)+"\_/"[k/n];k-=1
Provalo online!
Carboncino , 14 byte
←×_θ↖θ→↗θ×_θ‖M
Provalo online!
Correzione di +1 byte solo da ASCII.
Disegna metà dell'esagono e lo specchia.
C (gcc) , 119109 byte
a,i,j;f(n){for(i=a+=a=j=n*2;~j;)putchar(!i--?i=a,j--,13:i%(n*3)<n|j%(n*2)?(i-~j-n)%a?(i-j+n)%a?32:92:47:95);}
Provalo online!
- salvato 2 grazie a @ceilingcat
La tabella seguente non è aggiornata, quindi i valori possono differire, ma il concetto è:
- abbiamo 2 coppie di linee parallele
. . _______.______.________ | / \ | . . | / \ | . . | / \ |. | / \ | | \ / |. . | \ / | . . | \ / | __ | ___ \ ________ / ___ | ___ | . .
iteriamo x, y da dimensione a 0 e li sommiamo per verificare se a / deve essere stampato, sottraiamo per verificare \, usiamo modulo per controllare entrambi i paralleli.
i 65432109876543210. j i + jn ________ 8 13 + 7-4 => / \ 7 14 + 6-4 / \ 6 15 + 5-4 / \ 5 / \ 4 \ 1 + 3-4 => / 3 \ 2 + 2-4 / 2 \ 3 + 1-4 / 1 \ ________ / 0
Perl 5 ( -p
), 102 byte
s/_+/__$&/g,s/^|$/ /gm,s/^ *\S /$& /gm,s-( +)\\ -$&/ $1 \\ \\ $1 /
- for($\=' __ / \ \__/')x--$_}{
Provalo online!
-p
e }{
alla fine è un trucco per stampare solo il separatore del record di output $\
alla fine. Funziona solo per un record di input; l'intestazione viene utilizzata per stampare tutto in un collegamento tio.
$\=' __
/ \
\__/' # output record separator initialized with hexagon (size 1)
s/_+/__$&/g,s/^|$/ /gm,s/^ *\S /$& /gm,s-( +)\\ -$&/ $1 \\ \\ $1 /
- # regexes to increase the hexagon by 1
for .. --$_ # to repeat n-1 times where n is the input
Haskell , 150 byte
r=reverse
m(o:c:k)=o:c:c:c:k++" "
f 1=[["\\ /","__ "],["/__\\"]]
f n|[w:i,j]<-map m<$>f(n-1),_:_:k<-r$m w=[r k:w:i,k:j]
h[i,j]=unlines$r<$>r i++j
h.f
Provalo online!
Una versione più ricorsiva di questa risposta. L'arbitrarietà di questa sfida lo rende piuttosto frustrante.
Spiegazione
Questa risposta è un po 'difficile da spiegare. La sfida è, come ho già detto, arbitraria in alcuni modi, quindi il codice è una specie di semplice nido di simboli.
Idea
L'idea del programma qui è costruire le due metà. Cioè quando calcoliamo l'ennesimo esagono otteniamo le due metà per l'n-1esimo esagono e le usiamo per creare il successivo più grande.
Ci sono però alcuni avvertimenti. Costruiamo la metà superiore capovolta e costruiamo entrambe le metà specchiate da sinistra a destra. Lo facciamo perché è conveniente farlo in questo modo. Nessuna ragione profonda che rende le cose più brevi anche se rende le cose un po 'incomprensibili.
Dettagli
La prima riga è piuttosto semplice r
è un alias per reverse
. La seconda riga non è così semplice. m
è una funzione senza senso, esiste perché essa o un'operazione simile deve essere eseguita in alcuni punti. Non ha davvero un significato semantico. La migliore spiegazione di ciò che fa qui è il codice.
m(o:c:k)=o:c:c:c:k++" "
Da qui iniziamo ad arrivare a f
chi gestisce sostanzialmente tutta la logica. Il primo caso f
è il case base, è piuttosto standard
f 1=[["\\ /","__ "],["/__\\"]]
Nota che restituiamo un elenco di due elementi invece di una tupla. In qualsiasi programma sano utilizzeremo una tupla poiché è fissata a 2 elementi. Comunque in seguito mapperemo entrambi gli argomenti di questo con la stessa funzione. È difficile farlo con una tupla ma facile con un elenco, e l'elenco non presenta alcun inconveniente, quindi lo usiamo.
Poi abbiamo il caso induttivo. Per prima cosa recuperiamo il caso precedente e facciamo una doppia mappa m
su di esso. Questo rende l'esagono di 1 unità più largo (2 caratteri) e lo sposta di mezza unità (1 carattere) a destra (sebbene poiché l'intera cosa è al contrario, i caratteri spazio vengono aggiunti a destra ). Facciamo corrispondere questo modello a [w:i,j]
perché vogliamo usarlo w
per creare nuove righe in seguito. A proposito di quale prossimo facciamo le righe. Lo facciamo con una corrispondenza di pattern:
_:_:k<-r$m w
Questa è una specie di codice senza senso. Semplicemente mette insieme le cose che avevamo già per produrre l'output corretto. k
e il suo inverso forma le nuove righe, quindi le aggiungiamo. e la restituiamo.
Dopo f
abbiamo h
che trasforma l'output di f
in una stringa. Annulla tutte le stravaganti trasformazioni che abbiamo usato durante la costruzione e lo impacchetta per essere utilizzato.
Con tutto ciò ci limitiamo a comporre f
e h
per la funzione finale.
Haskell , 100 byte
f n=unlines[q<$>[1..3*n]++[1-n..0]|y<-[-n..n],let q x|abs y==n,x>n='_'|x==y='\\'|x+y==1='/'|1>0=' ']
Provalo online!
La risposta del golf AZTECCO più alcune nuove tecniche.
L'idea principale è che l'esagono è più semplice se trapiantiamo le prime n
colonne all'estremità.
|-|
______
/ \
/ \
/ \
\ /
\ /
\______/
|-|
______
\ /
\ /
\/
/\
/ \
______/ \
Ora tutti i /
e \
sono in una singola riga e _
sono tutti a sinistra di quelli. Ciò rende molto più semplice eseguire la strategia di AZTECCO per determinare il personaggio dalle coordinate. Per implementare queste coordinate rietichettate, sostituiamo le x
coordinate -con [1..4*n]
una versione ciclata e spostata [1..3*n]++[1-n..0]
.
Retina 0.8.2 , 94 byte
.+
$* ¶$&$* \G ¶$%'/$`$%_$%_$`\
r` \G
$%`\$'$%_$%_$%'/¶ ^¶( *) $1 $.&$*_$.&$*_$& T` `\_` +/$
Provalo online! Il collegamento include casi di test. Spiegazione:
.+
$* ¶$&$*
Inserisci due righe di n
spazi.
\G
¶$%'/$`$%_$%_$`\
Converti la prima riga nei lati superiori dell'esagono.
r` \G
$%`\$'$%_$%_$%'/¶
Converti la seconda riga nei lati inferiori dell'esagono.
^¶( *)
$1 $.&$*_$.&$*_$&
Inserisci la riga superiore.
T` `\_` +/$
Sostituisci la linea di fondo.
Python 2 , 119114 byte
i=n=input()
d=0
exec"k=i/n|d;print' '*i+'\_/'[~k]+'_ '[i-d<n]*2*(2*n+~i)+'\_/'[k]\nif i==d:d=i=-1\ni-=d|1;"*(n-~n)
Provalo online!
JavaScript (ES6), 109 107 byte
w=>(x=0,W=w*4,i=g=y=>~y?`
/\\_`[x++-W?y*!i|w/x|x>w*3?(x+~y+w)%W?(x+y)%W-w?1:3:2:4:x=i=0&y--]+g(y):'')(w*2)
Provalo online!
Commentato
w => ( // w = input
x = 0, // initialize x to 0
W = w * 4, // W = total width
i = // initialize i to a non-zero value
g = y => // g is a recursive function taking y
~y ? // if y is not equal to -1:
`\n /\\_`[ // list of characters
x++ - W ? // if this is not the end of the row:
y * !i | // if this is neither the first nor the last row
w / x | // or x is less than or equal to w
x > w * 3 ? // or x is greater than w * 3:
(x + ~y + w) % W ? // if (x - y - 1 + w) mod W is not equal to 0:
(x + y) % W - w ? // if (x + y) mod W is not equal to w:
1 // draw a space
: // else:
3 // draw a '\'
: // else:
2 // draw a '/'
: // else:
4 // draw a '_'
: // else:
x = i = 0 & y-- // decrement y, set x and i to 0 and draw a linefeed
] + g(y) // append the result of a recursive call
: // else:
'' // stop the recursion
)(w * 2) // initial call to g with y = w * 2
C (gcc) , 194 \$\cdots\$ 149 144 byte
Risparmiati 13 14 19 byte grazie a Ceilingcat !!!
p(n,c){for(;n--;)printf(L"/\\ _\n"+c);}i;t;f(n){p(n,2);for(i=t=p(2*n,3);i>=p(1,4);t=i/n?--i,1:t)i+=!p(!p(n+i<<!p(!p(n+~i,2),t),t&!i|2),!t)-2*t;}
Provalo online!
Spiegazione (prima di alcuni golf)
p(n,c){for(;n--;) // Helper function to print
putchar("/\\ _\n"[c]);} // one of '/', '\', ' ', '_' , or
// newline n times, this function
// also returns 0
i;t;f(n){ // Main function prints an n hexagon
p(n,2); // Print n leading spaces for the 1st
// line
for( // Main loop
i=t=p(2*n,3); // Set i and t to 0,
// and print 2*n '_'s for the 1st line
i>=p(1,4); // Loop until i goes below 0, and
// print a newline
// At the end of each loop:
i+=1-2*t, // increment i for the 1st half
// and then decrement i in the 2nd
t=i/n?--i,1:t) // keep t as t unless i equals n,
// then make t 1 and decrement i
// In the main loop:
p(n+~i,2), // print n-i-1 leading spaces
p(1,t), // print a '/' in the 1st half and a
// '\' in the 2nd
p(n+i<<1,t&!i|2), // print the 2*(n+i) middle spaces
// unless at the bottom print '_'s
p(1,!t); // print a '\' in the 1st half and a
// '/' in the 2nd
}
05AB1E , 33 29 25 byte
-4 ( 7 ) byte grazie a Kevin Cruijssen !
L+<'/úíºI·'_ך»∊¶¡`ðs‡).c
Provalo online!
05AB1E ha un incorporato tela che potrebbe essere utile, ma è abbastanza difficile da ottenere di lavoro, questo è solo l'comandi incorporati specchio º
e ∊
ed il builtin centralizzare .c
.
J , 50 47 byte
-3 grazie a Jonah!
' \/_'{~]|."1((0,]+2*|.)@=@i.,.3,3,~0$~<:,])@+:
Provalo online!
((0,]+2*|.)@=@i. ,. 3,3,~0$~<:,])@+:
0 0 0 0 3 3 3 3
1 0 0 2 0 0 0 0
0 1 2 0 0 0 0 0
0 2 1 0 0 0 0 0
2 0 0 1 3 3 3 3
]|."1
0 0 3 3 3 3 0 0
0 2 0 0 0 0 1 0
2 0 0 0 0 0 0 1
1 0 0 0 0 0 0 2
0 1 3 3 3 3 2 0
' \/_'{~
____
/ \
/ \
\ /
\____/
Python 3.8 (pre-rilascio) , 174 byte
n=int(input())
b,a,s="\/ "
z,f=range(n),lambda c,d,t:((n-1-i)*s+c+2*(n+i)*s+d for i in t)
print(f"{'_'*2*n:^{4*n}}",*f(a,b,z),*f(b,a,z[:0:-1]),f"{b:>{n}}{'_'*2*n}/",sep="\n")
Provalo online!
-1 byte grazie a @Duncan
-8 byte grazie a @Danis
Haskell , 129120 byte
g=mod
f n|a<-n*4=[c|y<-[-n..n],x<-[0..a],let c|x<1='\n'|g x(n*3+1)>n,abs y==n='_'|g(x+y)a==1='/'|g(x-y)a<1='\\'|1>0=' ']
Provalo online!
salvato 9 grazie a @Wheat Wizard e @xnor.
Controlla la risposta @xnor e il suo grande miglioramento nell'approccio al problema (evitando modulo e linee parallele)!
Equivalente alla mia risposta C.
Creiamo coordinate cartesiane usando la comprensione delle liste | y <- [0..n * 2], x <- [0..a]
[c | ..., sia c | ... | ... | ...] e restituiamo il carattere necessario in base a xy o x + y per disegnare linee ..
Usiamo modulo per disegnare linee multiple (2)
Il caso speciale è per _
cui non è necessario un rapporto ax / y ma un intervallo usato modulo per fare un solo confronto >
invece di un a> x> b simile
Ruby , 141 byte
->(n,g=->c,d{(1..n).map{|i|" "*(n-i)+d+" "*2*(n+i-1)+c}},l=g[?/,e=?\\].reverse){[" "*n+?_*n*2,g[e,?/],l[0..-2],l[-1].sub(/ +(?=\/)/,?_*n*2)]}
Provalo online!
Se è consentito un carattere di sottolineatura superiore, ci vorranno un po 'meno di byte, mentre l'esagono è carino come prima ;-)
Ruby , 114 byte
->n{[" "*n+?_*2*n,(g=->c,d{(1..n).map{|i|" "*(n-i)+c+" "*2*(n+i-1)+d}})[?/,e=?\\],g[e,?/].reverse," "*n+?‾*n*2]}
Provalo online!
05AB1E , 70 byte
ðש'_¹·×«©¶«©¹'/1Λ¹·Ì'.2Λ¹'\3.Λ«©„./`.;©¶¡Â‚€»`s'.ð:,¹3*>(£„._`:„/\‡,
Provalo online!
Perl 5 , 143 byte
sub f{$n=pop;@b=map{join'',$"x($n-$_),'/',' 'x($n+$_-1),'\\',$/}1..$n;join('',$"x$n,$u=__ x$n,$/,@b,map y|/\\|\\/|r,reverse@b)=~s| +/$|$u/|sr}
Provalo online!
sub f{
$n=pop; #n = input
@b=map{ #set array @b to line 2 - n+1
join'', #join parts into a line string
$" x ($n-$_), #space times n-1 '/', #char / ' ' x ($n+$_-1), #space times n+iterator minus 1 '\\', #char \ $/ #char newline
} 1..$n; #n lines join('', #return string of these joined: $" x $n, #n spaces $u = __ x$n, #n*2 underscores $/, #char newline
@b, #lines 2 to n+1 constructed above
map y|/\\|\\/|r, #and bottom part which is the same
reverse@b #as top part reversed and /\ rotated
)=~s| +/$|$u/|sr #and change last spaces of last line to _'s
}
Haskell ,
186168
byte
-18 byte con correzioni di xnor e rimozione di spazi non necessari sull'ultima riga
s=' '
(!)=replicate
e=reverse
h n=unlines$(\m->(n!s++(2*n)!'_'++n!s):e(e<$>m)++init m++[(n-1)!s++'\\':(2*n)!'_'++"/"])$(\i->i!s++'\\':(4*n-2*i-2)!s++'/':i!s)<$>[0..n-1]
Provalo online!
Ungolfed:
hex :: Int -> String
hex n = unlines $ first: middle ++ (init $ reverse (map reverse middle)) ++ [last]
where
first = replicate n ' ' ++ replicate (2*n) '_' ++ replicate n ' '
f i = replicate i ' ' ++ "/" ++ replicate (n-i + 2*n + n-i -2) ' ' ++ "\\" ++ replicate i ' '
middle = map f [n-1,n-2..0]
last = replicate (n-1) ' ' ++ "\\" ++ replicate (2*n) '_' ++ "/" ++ replicate (n-2) ' '