Arte ASCII del día # 3 - Santuarios chinos

16

En el episodio de hoy de AAOD, vamos a construir un santuario chino de diferentes alturas.

Considere los siguientes ejemplos para height ( N) 1to6

N = 1:

       .
       |
  .   ]#[   .
   \_______/
.    ]###[    .
 \__]#.-.#[__/
  |___| |___|
  |___|_|___|
  ####/_\####
     |___|
    /_____\

N = 2:

         .
         |
    .   ]#[   .
     \_______/
  .    ]###[    .
   \___________/
.     ]#####[     .
 \___]#.---.#[___/
  |__|_|   |_|__|
  |__|_|___|_|__|
  #####/___\#####
      |_____|
     /_______\

N = 3:

           .
           |
      .   ]#[   .
       \_______/
    .    ]###[    .
     \___________/
  .     ]#####[     .
   \_______________/
.      ]#######[      .
 \____]#.-----.#[____/
  |__|__|     |__|__|
  |__|__|_____|__|__|
  ######/_____\######
       |_______|
      /_________\

N = 4:

             .
             |
        .   ]#[   .
         \_______/
      .    ]###[    .
       \___________/
    .     ]#####[     .
     \_______________/
  .      ]#######[      .
   \___________________/
.       ]#########[       .
 \_____]##.-----.##[_____/
  |__|__|_|     |_|__|__|
  |__|__|_|_____|_|__|__|
  ########/_____\########
         |_______|
        /_________\

N = 5:

               .
               |
          .   ]#[   .
           \_______/
        .    ]###[    .
         \___________/
      .     ]#####[     .
       \_______________/
    .      ]#######[      .
     \___________________/
  .       ]#########[       .
   \_______________________/ 
.        ]###########[        .
 \______]###.-----.###[______/
  |__|__|___|     |___|__|__|
  |__|__|___|_____|___|__|__|
  ##########/_____\##########
           |_______|
          /_________\

N = 6:

                 .
                 |
            .   ]#[   .
             \_______/
          .    ]###[    .
           \___________/
        .     ]#####[     .
         \_______________/
      .      ]#######[      .
       \___________________/
    .       ]#########[       .
     \_______________________/ 
  .        ]###########[        .
   \___________________________/ 
.         ]#############[         .
 \_______]####.-----.####[_______/
  |__|__|__|__|     |__|__|__|__|
  |__|__|__|__|_____|__|__|__|__|
  ############/_____\############
             |_______|
            /_________\

y así.

Detalles de construcción

Estoy seguro de que la mayoría de los detalles sobre el patrón son claros. Aquí hay algunos detalles más finos:

  • La puerta en la parte inferior del santuario puede ser como mínimo de 1 _ancho y como máximo puede ser de 5 _ancho.
  • Siempre habrá dos .directamente encima de los pilares alrededor de la puerta (dos verticales |).
  • Las escaleras comienzan con el mismo ancho que la puerta y aumentan como se muestra en el patrón.
  • Los ]##..##[bloques sobre cada nivel de techo aumentan de tamaño de 2arriba a abajo.
  • Los \__...__/niveles de los techos aumentan de tamaño de 4arriba a abajo.
  • Los bloques de paredes alrededor de la puerta deben contener 1 _como mínimo y como máximo, 3 _entre los dos |. La prioridad va a los bloques de pared exterior para que el más cercano a la puerta tenga un tamaño variable para cada nivel.
  • El espacio entre el .y el ](o [) se llena #en el techo justo encima de las puertas.

Detalles del desafío

  • Escriba una función o programa completo que lea un entero positivo mayor que a 0través de STDIN / ARGV / argumento de función o equivalente más cercano y emite (a STDOUT o equivalente más cercano) el NSantuario chino
  • La nueva línea final es opcional.
  • No debe haber espacios finales o suficientes espacios finales para rellenar la salida en el rectángulo delimitador mínimo.
  • No debe haber espacios iniciales que no sean parte del patrón.

Tabla de clasificación

La primera publicación de la serie genera una tabla de clasificación.

Para asegurarse de que sus respuestas aparezcan, comience cada respuesta con un título, utilizando la siguiente plantilla de Markdown:

# Language Name, N bytes

¿Dónde Nestá el tamaño de su envío? Si mejora su puntaje, puede mantener los puntajes antiguos en el título, tachándolos. Por ejemplo:

# Ruby, <s>104</s> <s>101</s> 96 bytes
Optimizador
fuente
El ancho de la puerta me parece bastante arbitrario, ¿por qué es 1 en el N=1caso? ¿Por qué no 3 y tienen ventanas laterales más pequeñas como en el N=2caso?
Matty
Además, en el N=1caso, ¿no es el primer techo demasiado largo (ancho)?
Matty
@Matty con respecto a la puerta: si la puerta era de ancho 3, entonces no habría #al lado de la puerta .para apoyarla ]y [sobre ella. Acerca del tamaño del techo inicial: es el tamaño del techo en el techo superior de cada altura.
Optimizador
Estaba preguntando sobre el techo más bajo justo encima de las ventanas. En todos los demás casos, es el tamaño del techo sobre él +4 (+2 en ambos lados). Pero aquí está +8.
Matty
@ Matty, tienes razón. Fijo.
Optimizador

Respuestas:

2

CJam, 200 bytes

-4'.-4'|]2/ri:M),{2af.-~[W'.I3+~']'#I)*][-2'\'_I2*4+*]]}fI~[~M)<']
M2m1e>'#*'.'-M3e<:D*][-3"|__"M*M2+M2*(e>:L<"_|"D~][_~~'_*][-3'#L)*
'/'_D*][L2+~'|'_D)*][L)~'/'_D2+*]]{{_0<{~S*}&}%s_W%1>"\/]""/\["erN}%

Se agregaron nuevas líneas para evitar el desplazamiento. Pruébalo en línea

Breve explicacion:

El programa construye la mitad izquierda del santuario (incluido el centro), luego lo invierte y reemplaza algunos caracteres para obtener la mitad derecha. Una serie de n espacios se representa como el número ~ n (bit a bit "no") durante la construcción, y se reemplaza con los espacios reales al final.

El programa comienza con las 2 líneas superiores, luego para cada nivel de techo, antepone todas las líneas anteriores con 2 espacios y agrega el nuevo techo (2 líneas). El último techo se modifica para agregar la parte de "puerta de arriba".

Luego, el muro superior se construye repitiendo "| __" y truncando a la longitud correcta, seguido de un "_ |" fijo y espacios. La pared se duplica y los espacios de las puertas se reemplazan con guiones bajos. Finalmente, la parte inferior se construye línea por línea.

aditsu
fuente
5

Perl, 332 316 294

$:=($w=<>)*2+6;$r=2x($m=$w>3?3:$w);$k=1x($w-3).b.4x$m;
y!a-f1-4!/.|\\[,#_ -!,/,/,s/(.*),(.*).{@{-}}/$2$1/,printf"%$:s%s
",y!/\\[!\\/]!r,(reverse=~s/.//r)for@x=(b,c,
(map{b33.3x$_.e.1x$_,"[#$k,"x/$w/.a__.22x$_}1..++$w),
_c.3x$m.f.($z=substr"|__"x$:,0,2*++$w),"_|$r,$z","d$r,".11x$w,c_.$r,d__.$r)

Trate de mí .

C, 371

d,i,w;char s[1<<24];m(){v(w,13);}p(){puts(s+1);}
v(i,j){s[w-i]=".|]\\#/"[j%7];s[w+i]=".|[/#\\"[j%7];
while(i--)s[w-i]=s[w+i]="# _-"[j/7];}
main(l){scanf("%d",&l);d=l>3?3:l;m(w=l*2+6);p(v(0,0));
for(v(0,1);i++<=l;v(i*2+2,17))p(),v(i*2+3,7),m(p(v(i,2)));v(l+2,2);p(v(d,21));
for(m(i=w-3);i>d+1;i-=3)v(i,15);p(v(d,8));p(v(d,15));
v(w-3,4);m(p(v(d,19)));p(v(d+1,15));p(v(d+2,19));}

Trate de mí .

JavaScript, 365

Lo anterior se puede traducir casi de 1 a 1 en JavaScript:

s=[];r="";i=0;m=()=>v(w,13);p=()=>r+=s.join('')+"\n";
v=(i,j)=>{s[w-i]=".|]\\#/"[j%7];s[w+i]=".|[/#\\"[j%7];
while(i--)s[w-i]=s[w+i]="# _-"[j/7|0];};
f=l=>{d=l>3?3:l;m(w=l*2+6);p(v(0,0));
for(v(0,1);i++<=l;v(i*2+2,17))p(),v(i*2+3,7),m(p(v(i,2)));v(l+2,2);p(v(d,21));
for(m(i=w-3);i>d+1;i-=3)v(i,15);p(v(d,8));p(v(d,15));
v(w-3,4);m(p(v(d,19)));p(v(d+1,15));p(v(d+2,19));}

Utilizar:

f(2);console.log(r)
nutki
fuente
La versión C se bloquea para tamaños superiores a 12. Dado que parece que está utilizando una cadena de tamaño fijo para contener resultados temporales, creo que siempre tendrá un límite superior, sin importar cómo elija el tamaño s. A menos que lo asigne dinámicamente, por supuesto.
Reto Koradi
@RetoKoradi, Sí, tienes razón, publiqué la versión con un pequeño buffer por error. Pero al final, a menos que se asigne dinámicamente, siempre habrá un límite.
nutki
4

Python 2, 356 352 347 344 bytes

n=input()
A,B,C,D,E,F,G,H,I='_ |\/#.]['
def p(*S):
 for s in S:print(5+2*n-len(s)/2)*B+s
p(G,C,'.   ]#[   .')
for i in range(n):b=B*(4+i);p(D+A*(7+4*i)+E,G+b+H+F*(3+2*i)+I+b+G)
d=2*min(3,n)-1
a=A*(2+i)
f=F*(1+i-d/2)
j=4+2*i-d/2
w=('|__'*n)[:j-1]+A+C
v=w[::-1]
p(D+a+H+f+G+'-'*d+G+f+I+a+E,w+B*d+v,w+A*d+v,F*j+E+A*d+D+F*j,C+A*(d+2)+C,E+A*(d+4)+D)

Esto básicamente construye el santuario línea por línea. La función pimprime una cadena con los espacios necesarios para centrarla.

Utilicé python 2 para guardar muchos bytes, porque el objeto de mapa de python 3 no se dispara. Supongo que siempre debería jugar golf en Python 2 , solo guarda unos pocos bytes más (incluso si es solo para no tener que analizar la entrada a int). Jejeje, no es como el código es bonito para jugar golf en primer lugar.

Editar: y, por supuesto, ya no necesito el mapa ...

Aquí está el código en forma no golfizada:

n = int(input())

# A function to print strings centered
half_width = 5 + 2*n
def p(string):
    spaces = ' ' * (half_width - len(string) // 2)
    print(spaces + string)

# The rooftops
p('.')
p('|')
p('.   ]#[   .')
for i in range(n):
    p('\\' + '_'*(7 + 4*i) + '/')
    p('.{0}]{1}[{0}.'.format(' '*(i + 4), '#'*(3 + 2*i)))

# The bottom rooftop
door_width = 2 * min(3, n) - 1
# (11+4i - (3+2i) - 4) / 2 = (4 + 2i) / 2 = 2 + i
p('\{0}]{1}.{2}.{1}[{0}/'.format('_'*(2 + i), '#'*(1 + i - door_width // 2), '-'*door_width))

# The windows
w = '|__'*n
w = w[:4 + 2*i  - door_width // 2]
if w[-1] == '|':
    w = w[:-1] + '_'
w += '|'
p(w + ' '*door_width + w[::-1])
p(w + '_'*door_width + w[::-1])

# The foundation and the stairs
w = '#'*(4 + 2*i - door_width // 2)
p(w + '/' + '_'*(door_width) + '\\' + w)

# The remaining stairs
p('|' + '_'*(door_width + 2) + '|')
p('/' + '_'*(door_width + 4) + '\\')
Matty
fuente
1
Como está usando Python 2, puede cambiar print(B*(5+2*n-len(s)//2)+s)a print B*(5+2*n-len(s)/2)+s(eliminar paréntesis y cambiar //a /).
user12205
1
Gracias @ace, no sabía que python2 ignoraba la división de flotación.
Matty
1
@ Matty No ignora la división de flotación. Estás realizando la división en enteros , por lo que el resultado es un entero. Solo hace división flotante si uno o más operandos es flotante.
mbomb007
2
Si reorganiza el orden de print B*(5+2*n-len(s)/2)+sa print(5+2*n-len(s)/2)*B+s, puede eliminar el espacio después print.
isaacg
4

JavaScript ( ES6 ), 440

Editar error de dintel fijo

Una función con altura como parámetro, salida a la consola.

Al usar mucho la cadena de plantilla , todas las nuevas líneas son significativas y cuentan.

Ejecute el fragmento para probar en Firefox (con salida de consola)

f=x=>{R=(n,s=0)=>' #_-'[s][Z='repeat'](n),M=c=>R(2)+'|__'[Z](z+1).slice(0,z-1)+'_|'+R(y,c)+'|_'+'__|'[Z](z+1).slice(1-z)
for(z=x+x+(x<2)+(x<3),y=x>2?5:x>1?3:1,l=-1,o=`${t=R(x+x+5)}.
${t}|
`;l++<x;)o+=`${t=R(x+x-l-l)}.${u=R(l+3)}]${R(l*2+1,1)}[${u}.
 ${t}\\${l-x?R(7+l*4,2):`${t=R(x+1,2)}]${u=R(x<3||x-2,1)}.${R(y,3)}.${u}[${t}`}/
`;console.log(`${o+M(0)}
${M(2)}
${R(2)}${t=R(z,1)}/${u=R(y,2)}\\${t}
${R(1+z)}|__${u}|
${R(z)}/____${u}\\`)}

// TEST
f(1),f(2),f(3),f(4),f(5),f(6)
Output 1 to 6 in console

Versión sin golf para prueba interactiva:

// Not so golfed

f=x=>{
  R=(n,s=0)=>' #_-'[s].repeat(n); // base building blocks
  M=c=>R(2)+'|__'.repeat(z+1).slice(0,z-1)+'_|'+R(y,c)+'|_'+'__|'.repeat(z+1).slice(1-z); // manage door level

  z=x+x+(x<2)+(x<3); // door and stairs surroundings
  y=x>2?5:x>1?3:1; // door and stairs width
  
  o = `${R(x+x+5)}.\n${R(x+x+5)}|\n`; // top 
  for(l=-1;l++<x;)
    o += `${ // even row
      t=R(x+x-l-l) // left padding
    }.${
      u=R(l+3)
    }]${
      R(l*2+1,1)
    }[${
      u
    }.\n ${ // end even row, start odd row
      t // left padding
    }\\${
      l-x?R(7+l*4,2)
      :`${t=R(x+1,2)}]${u=R(x<3||x-2,1)}.${R(y,3)}.${u}[${t}` // if last row before the door, insert lintel 
    }/\n`;
  
  o += `${
    M(0) // door level row 1
  }\n${
    M(2) // door level row 2
  }\n${
    R(2)}${t=R(z,1)}/${u=R(y,2)}\\${t  // stairs row 1
  }\n${ 
    R(1+z)}|__${u  // stairs row 2
  }|\n${ 
    R(z)}/____${u // stairs row 3
  }\\`;
  
  out(o)
}

out=x=>O.innerHTML=x

f(3)
<input id=I value=3><button onclick='f(+I.value)'>-></button><br>
<pre id=O></pre>

edc65
fuente
2

Haskell, 473 bytes

g '\\'='/'
g '/'='\\'
g '['=']'
g ']'='['
g x=x
z=reverse
d=min 3
m n s=putStrLn$(6+2*n+1-length s)#' '++map g(z s)++tail s
n#c=replicate n c
t 0=".";t n=n#'#'++"["++(2+n)#' '++"."
r 0="|";r n=(2+2*n)#'_'++"/"
q n=d n#'-'++"."++max(n-2)1#'#'++"["++(n+1)#'_'++"/"
e i n=d n#i++"|_"++z(take(max(n+2)(2*n-1))(cycle"|__"))
b n=d n#'_'++"\\"++max(2*n)(3+n)#'#'
w i n=(d n+1+i)#'_'++["|","\\"]!!i
f n=mapM_(m n)$[u k|k<-[0..n],u<-[t,r]]++(t(n+1):map($n)[q,e ' ',e '_',b,w 0,w 1])
Damien
fuente
OK gracias. Parece correcto ahora
Damien
Creo que está bien ahora. por cierto, la base en tu santuario N = 1 tiene un # más
Damien
Sí, funciona ahora. Se corrigió ese error adicional no intencionado#
optimizador
0

C, 660 bytes

El patrón simplemente parecía ser demasiado irregular para encontrar algo elegante, particularmente en un lenguaje sin procesamiento de cadenas. Así que aquí está mi enfoque de fuerza bruta:

m,d,w,k,j;r(n,c){for(j=0;j++<n;)putchar(c);}c(char*s){for(;*s;)putchar(*s++);}f(n){m=n*2;d=n<3?m-1:5;w=m-d/2+2;r(m+5,32);c(".\n");r(m+5,32);c("|\n");for(;;){r(m-k*2,32);c(".");r(k+3,32);c("]");r(k*2+1,35);c("[");r(k+3,32);c(".\n");if(k==n)break;r(m-k*2+1,32);c("\\");r(k++*4+7,95);c("/\n");}c(" \\");r(n+1,95);c("]");r(n-d/2,35);c(".");r(d,45);c(".");r(n-d/2,35);c("[");r(n+1,95);c("/\n");for(k=0;k<2;){c("  ");for(j=0;j<w/3;++j)c("|__");c("|_|"-w%3+2);r(d,k++?95:32);c(!w%3?"|":w%3<2?"|_":"|_|");for(j=0;j<w/3;++j)c("__|");c("\n");}c("  ");r(w,35);c("/");r(d,95);c("\\");r(w,35);c("\n");r(w+1,32);c("|");r(d+2,95);c("|\n");r(w,32);c("/");r(d+4,95);c("\\\n");}

Antes de jugar al golf:

#include <stdio.h>

void r(int n, int c) {
    for (int i = 0; i++ < n; )
        putchar(c);
}

void c(char* s) {
    for (; *s; ++s) putchar(*s);
}

int f(int n) {
    int m = n * 2;
    int d = n < 3 ? m - 1 : 5;
    int w = m - d / 2 + 2;

    r(m + 5, 32);
    c(".\n");

    r(m + 5, 32);
    c("|\n");

    for (int k = 0; ; ++k) {
        r(m - k * 2, 32);
        c(".");
        r(k + 3, 32);
        c("]");
        r(k * 2 + 1, 35);
        c("[");
        r(k + 3, 32);
        c(".\n");

        if (k == n) break;

        r(m - k * 2 + 1, 32);
        c("\\");
        r(k * 4 + 7, 95);
        c("/\n");
    }

    c(" \\");
    r(n + 1, 95);
    c("]");
    r(n - d / 2 , 35);
    c(".");
    r(d, 45);
    c(".");
    r(n - d / 2 , 35);
    c("[");
    r(n + 1, 95);
    c("/\n");

    for (int k = 0; k < 2; ++k) {
        c("  ");
        for (int j = 0; j < w / 3; ++j)
            c("|__");
        c("|_|" - w % 3 + 2);
        r(d, k ? 95 : 32);
        c(!w % 3 ? "|" : w % 3 < 2 ? "|_" : "|_|");
        for (int j = 0; j < w / 3; ++j)
            c("__|");
        c("\n");
    }

    c("  ");
    r(w, 35);
    c("/");
    r(d, 95);
    c("\\");
    r(w, 35);
    c("\n");

    r(w + 1, 32);
    c("|");
    r(d + 2, 95);
    c("|\n");

    r(w, 32);
    c("/");
    r(d + 4, 95);
    c("\\\n");

    return 0;
}

No hay mucho que explicar aquí. Simplemente va línea por línea y genera el recuento necesario de cada personaje. Traté de mantener el código compacto, pero todavía suma. des el ancho de la puerta, wel ancho de cada pared de ladrillos.

Reto Koradi
fuente
1
c(char*s){for(;*s;)putchar(*s++);}==> #define c printf; r(n,c){for(j=0;j++<n;)putchar(c);}==>r(n,C){while(n--)putchar(C);}
usuario12205