Argyle ASCII Art

27

Écrivez un programme (ou une fonction) qui accepte un entier positif.

Si l'entrée est 1, imprimez (ou retournez) deux diamants voisins côte à côte, chacun avec une longueur de côté d'une barre oblique:

/\/\
\/\/

Pour chaque entrée Nsupérieure à 1, regardez la sortie pour N-1et pour chaque paire de diamants voisins, insérez un nouveau diamant entre eux dont la longueur latérale est la somme des longueurs latérales des deux voisins. Imprimez (ou retournez) ce nouveau motif de diamant.

Donc, quand 2est entrée, nous regardons la sortie 1et pouvons voir qu'il y a deux diamants voisins, tous deux de longueur latérale 1. Donc, nous insérons un diamant de longueur latérale 2 (1 + 1) entre eux:

   /\
/\/  \/\
\/\  /\/
   \/

Pour l'entrée, 3nous regardons la sortie 2et ajoutons deux diamants de longueur latérale 3 (1 + 2 et 2 + 1) entre les deux paires de diamants voisins:

    /\        /\
   /  \  /\  /  \
/\/    \/  \/    \/\
\/\    /\  /\    /\/
   \  /  \/  \  /
    \/        \/

Poursuivant le modèle, la sortie pour 4est:

                    /\            /\
     /\            /  \          /  \            /\
    /  \    /\    /    \        /    \    /\    /  \
   /    \  /  \  /      \  /\  /      \  /  \  /    \
/\/      \/    \/        \/  \/        \/    \/      \/\
\/\      /\    /\        /\  /\        /\    /\      /\/
   \    /  \  /  \      /  \/  \      /  \  /  \    /
    \  /    \/    \    /        \    /    \/    \  /
     \/            \  /          \  /            \/
                    \/            \/

Etc.

Vos sorties peuvent avoir des espaces de fin sur toutes les lignes mais seulement jusqu'à un retour à la ligne de fin (et aucun retour à la ligne de début).

Le code le plus court en octets gagne.

Loisirs de Calvin
la source
1
OEIS pertinent: oeis.org/A002487 .
orlp

Réponses:

8

Pyth, 50 49 octets

L.rR"\/"_bjbyK.tsm+Jm+*\ k\\dyJu.iGsM.:G2tQjT9djK

Manifestation

Explication:

L.rR"\/"_bjbyK.tsm+Jm+*\ k\\dyJu.iGsM.:G2tQjT9djK
                                                     Implicit:
                                                     Q = eval(input())
                                                     T = 10
                                                     d = ' '
                                                     b = '\n'
L                                                    def y(b): return
 .rR"\/"                                             Swap \ and / in
        _b                                           reversed input.
                                                     This effectively vertically
                                                     mirrors the input.
                               u                     Apply the function repeatedly
                                           jT9       Starting with [1, 1]
                                         tQ          and repeating Q - 1 times
                                .iG                  interlace G (input) with
                                     .:G2            All 2 element substrings of G
                                   sM                mapped to their sums.
                 m                                   map over these values
                                                     implicitly cast to ranges
                    m       d                        map over the range values
                                                     impicitly cast to ranges
                     +*\ k\\                         to k spaces followed by
                                                     a backslash.
                   J                                 Save to J, which is roughly:
                                                     \
                                                      \
                  +          yJ                      And add on y(J), giving
                                                     \
                                                      \
                                                      /
                                                     /
                s                                    Combine the half diamonds
                                                     into one list.
              .t                              d      Traspose, filling with ' '.
             K                                       Save to K, giving
                                                     something like:
                                                     \  /
                                                      \/
            y                                        Vertically mirror.
          jb                                         Join on newlines and print.
                                               jK    Join K on (implicitly)
                                                     newlines and print.
isaacg
la source
1
Quelles sont les chances? J'ai aussi exactement ai u.iGsM.:G2tQjT9dans ma solution (partielle). Je n'ai jamais regardé votre réponse ...
orlp
2
@orlp Il n'y a souvent qu'une seule meilleure façon de faire quelque chose.
isaacg
5

Lisp commun, 425

(labels((a(n)(if(> n 1)(loop for(x y)on(a(1- n))by #'cdr collect x when y collect(+ x y))'(1 1))))(lambda(~ &aux(l(a ~))(h(apply'max l))(w(*(apply'+ l)2))(o(* 2 h))(m(make-array(list o w):initial-element #\ ))(x 0)(y h))(labels((k(^ v)(setf(aref m y x)^(aref m(- o y 1)x)v)(incf x))(d(i)(when(plusp i)(k #\\ #\/)(incf y)(d(1- i))(decf y)(k #\/ #\\))))(mapc #'d l))(dotimes(j o)(fresh-line)(dotimes(i w)(princ(aref m j i))))))

Exemple

(funcall *fun* 4)

                    /\            /\                    
     /\            /  \          /  \            /\     
    /  \    /\    /    \        /    \    /\    /  \    
   /    \  /  \  /      \  /\  /      \  /  \  /    \   
/\/      \/    \/        \/  \/        \/    \/      \/\
\/\      /\    /\        /\  /\        /\    /\      /\/
   \    /  \  /  \      /  \/  \      /  \  /  \    /   
    \  /    \/    \    /        \    /    \/    \  /    
     \/            \  /          \  /            \/     
                    \/            \/                    

Non golfé

(labels
    ((sequence (n)
       (if (> n 1)
           (loop for(x y) on (sequence (1- n)) by #'cdr
                 collect x
                 when y
                   collect(+ x y))
           '(1 1))))
  (defun argyle (input &aux
                  (list (sequence input))
                  (half-height (apply'max list))
                  (width (* (apply '+ list) 2))
                  (height (* 2 half-height))
                  (board (make-array
                          (list height width)
                          :initial-element #\ ))
                  (x 0)
                  (y half-height))
    (labels ((engrave (^ v)
               (setf (aref board y              x) ^ ;; draw UP character
                     (aref board (- height y 1) x) v ;; draw DOWN character (mirrored)
                     )
               (incf x) ;; advance x
               )
             (draw (i)
               (when (plusp i)
                 (engrave #\\ #\/)  ;; write opening "<" shape of diamond
                 (incf y)
                 (draw (1- i))   ;; recursive draw
                 (decf y)
                 (engrave #\/ #\\)  ;; write closing ">" shape of diamond
                 )))
      ;; draw into board for each entry in the sequence
      (mapc #'draw list))

    ;; ACTUAL drawing
    (dotimes(j height)
      (fresh-line)
      (dotimes(i width)
        (princ (aref board j i))))
    board))
coredump
la source
3

CJam, 59 58 57 octets

YXbri({{_2$+\}*]}*:,_:|,S*0'\tf{fm>_W%'\f/'/f*}:+zN*_W%N@

Merci à @ MartinBüttner d'avoir joué au golf sur 1 octet.

Essayez-le en ligne dans l' interpréteur CJam .

Idée

Pour l'entrée 3 , par exemple, nous générons

\  
/  
\  
 \ 
  \
  /
 / 
/  
\  
 \ 
 / 
/  
\  
 \ 
  \
  /
 / 
/  
\  
/  

en faisant tourner la chaîne  \et en remplaçant certaines barres obliques inverses par des barres obliques.

Ensuite, nous compressons le tableau résultant (transposons les lignes et les colonnes) pour obtenir la moitié inférieure de la sortie souhaitée.

La moitié supérieure est octet par octet égale à la moitié inférieure en sens inverse.

Code

YXb     e# Push A := [1 1] and 2 in unary.
ri(     e# Read an integer fro STDIN and subtract 1.
{       e# Do the following that many times:
  {     e#   For each I in A but the first:
    _2$ e#     Push a copy of I and the preceding array element.
    +\  e#     Compute the sum of the copies and swap it with I.
  }*    e#
  ]     e#   Collect the entire stack in an array.
}*      e#
:,      e# Replace each I in A with [0 ... I-1].
_       e# Push a copy of A.
:|      e# Perform set union of all the ranges.
,S*     e# Get the length (highest I in A) and push a string of that many spaces.
0'\t    e# Replace the first space with a backslash.
f{      e# For each range in A, push the generated string; then:
  fm>   e#   Rotate the string by each amount in the array.
  _W%   e#   Push a reversed copy of the resulting array of strings.
  '\f/  e#   In each string, split at backslashes.
  '/f*  e#   Join each string, separating with slashes.
}       e#
:+      e# Concatenate the resulting arrays of strings.
zN*     e# Zip and join, separating by linefeeds.
_W%     e# Push a reversed copy of the string.
N@      e# Push a linefeed and rotate the original string on top of it.
Dennis
la source
1

Rév.1: Ruby 170

Nouvelle méthode évitant de créer le gros diamant et de le réduire.

->n{a=[1]
m=1<<n-1
(m-1).times{|i|a<<a[i]<<a[i]+a[i+1]}
(-b=a.max).upto(b-1){|j|0.upto(m){|i|d=' '*q=a[-i]*2
(j*2+1).abs<q&&(d[j%q]=?\\;d[-1-j%q]=?/)   
print d}
puts""}}

Rév 0: Ruby, 187

->n{a=[1]
m=1<<n-1
(m-1).times{|i|a<<a[i]<<a[i]+a[i+1]}
(2*b=a.max).times{|j|
0.upto(m){|i|d=' '*b*2;d[(b+j)%(b*2)]='\\';d[(b-1-j)%(b*2)]=?/
r=b-a[-i]
d.slice!(b-r,r*2)
print d}
puts ""}}

Les tailles des diamants sont calculées conformément à la relation de récurrence de https://oeis.org/A002487 Ainsi, nous faisons le tableau acontenant tous les éléments pour toutes les lignes de 1 à n. Nous ne sommes intéressés que par les derniers 1<<n-1éléments (Ruby nous permet de les obtenir du tableau en utilisant des index négatifs, -1 étant le dernier élément du tableau), plus un inital à 1partir de la position 0.

Ligne par ligne et diamant par diamant, nous dessinons la ligne de caractères pour le plus gros diamant, puis découpons les colonnes du milieu pour obtenir la ligne du diamant requis. Rev 1 est plus court, mais j'ai aimé cette méthode.

L'arithmétique modulaire est utilisée pour boucler autour de sorte que la même expression ajoute tout /directement et de même une expression ajoute tout \directement.

Non testé dans le programme de test

f=->n{
  a=[1]
  m=1<<n-1
  (m-1).times{|i|a<<a[i]<<a[i]+a[i+1]}                   #concatenate a[i] and a[i]+a[i+1] to the end of a
    (2*b=a.max).times{|j|                                #run through lines (twice the largest number in a
      0.upto(m){|i|                                      #run through an initial '1' plus the last m numbers in a
      d=' '*b*2;d[(b+j)%(b*2)]='\\';d[(b-1-j)%(b*2)]=?/  #d is the correct string for this line of the largest diamond
      r=b-a[-i]                                          #calculate number of characters to be deleted from middle of d
      d.slice!(b-r,r*2)                                  #and delete them
      print d                                            #print the result
    }
    puts ""                                              #at the end of the line, print a newline
  }
}

f.call(gets.to_i)
Level River St
la source