Crea un rompecabezas de nonogramas

24

Un nonograma es un rompecabezas lógico bidimensional que se parece a esto (capturas de pantalla del juego Pixelo , mi juego de nonogramas favorito):

Un tablero de nonogramas vacío

El objetivo del juego es descubrir qué imagen codifican esos números. Las reglas son simples: un número en una columna o fila significa que en algún lugar de esa columna o fila, se rellenan muchos cuadros en una fila. Por ejemplo, la fila inferior de la imagen de arriba no debe tener cuadros rellenados, mientras que el de arriba debe tener todos sus cuadros llenos. La tercera fila desde abajo tiene 8 cajas llenas, y todas estarán en una fila.

Dos o más números para la misma columna o fila significan que hay múltiples "tiradas" de cajas llenas, con al menos un espacio entre ellas, con esas longitudes. El orden se conserva. Por ejemplo, hay tres cuadros rellenados en la columna de la derecha de la imagen de arriba, al menos un espacio debajo de ellos, y luego un cuadro más rellenado.

Aquí está el mismo rompecabezas, casi completado:

Un tablero de nonogramas casi terminado

(Las X no son importantes, son solo una pista que el jugador deja para sí mismo para decir "Esta casilla definitivamente no está llena". Piensa en banderas en Buscaminas. No tienen reglas de significado).

Esperemos que pueda ver que, por ejemplo, las columnas del medio con sugerencias que dicen "2 2" tienen dos tramos de 2 longitudes de cajas rellenas.

Su misión, si elige aceptarla, es escribir un programa o función que cree un rompecabezas como este. Se le da el tamaño del tablero como un entero entero (5 <= n <= 50) en stdin o como argumento (no hay ninguna razón por la cual un rompecabezas de nonograma tiene que ser cuadrado, pero para este desafío lo será). Después de eso, se le dará una serie de 1s y 0s que representan cuadrados llenos y sin llenar en la imagen, respectivamente. Los primeros n de ellos son la fila superior, luego la siguiente fila, etc. Volverá o imprimirá para formar un tablero de 2 * 1 celdas (porque se ven mejor, y le da espacio para pistas de 2 dígitos para una columna ), todos ellos vacíos, con sugerencias correspondientes a los datos de entrada.

Formato de salida

Formato de salida

Muestra

Entrada:

./nonogram <<< '5 0 1 1 1 0 1 1 0 1 1 1 0 1 0 1 1 1 0 1 1 0 1 1 1 0'
                                 OR
      n(5,[0,1,1,1,0,1,1,0,1,1,1,0,1,0,1,1,1,0,1,1,0,1,1,1,0])

Imagen:

Primera imagen de ejemplo

Salida:

           1
         2 1 2
       3 2 1 2 3
     +----------
    3|
  2 2|
1 1 1|
  2 2|
    3|

Entrada:

./nonogram <<< '15 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 1 0 0 0 0 0 1 1 1 1 0 0 0 0 1 1 1 0 0 1 0 1 0 0 1 0 0 0 1 1 1 1 1 0 1 0 1 1 1 1 0 0 0 1 1 1 1 1 0 1 1 1 1 0 0 0 0 0 1 1 1 1 1 0 0 0 1 1 0 0 0 0 0 0 1 1 1 0 0 0 0 1 1 0 0 0 1 0 0 0 1 0 0 0 1 0 1 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1'

Imagen:

Segunda imagen de ejemplo

Salida:

                   1                       1
                 1 1 3       3 5   5 3   3 1
                 7 2 3 2 4 2 3 210 2 3 0 4 215
               +------------------------------
              2|
              1|
              1|
              1|
              1|
            1 1|
        3 3 1 1|
        1 5 1 1|
          3 5 3|
          1 5 1|
          1 3 1|
      1 1 1 1 1|
1 1 1 1 1 1 1 1|
           11 3|
           11 3|

Aclaraciones

  • Su salida no necesita ser un rompecabezas solucionable. No todos los nonogramas tienen solución, pero esa no es su preocupación. Simplemente envíe las sugerencias que corresponden a la entrada, ya sea que sean un buen rompecabezas o no.
  • Se permite un programa que tome argumentos en la línea de comando. Esto se afirma más arriba, pero es posible tener una idea equivocada. Para eso están las aclaraciones.
  • 0Es obligatorio imprimir una fila o columna que no haya rellenado los cuadros. No digo esto con palabras en ninguna parte, pero está en los datos de muestra.
metro subterráneo
fuente
Ya casi termino con mi solución. ¿Podemos suponer que no habrá un número de cuadros de 3 dígitos en una fila o columna?
Afuous
2
@voidpigeon: 5<=n<=50es la especificación, por lo que no puede haber ningún número de 3 dígitos
Kyle Kanos
Después de publicar esta pregunta, comencé a trabajar en una solución yo mismo. No lo publicaré todavía (según esta meta respuesta), pero publicaré mi recuento de bytes para que ustedes tengan algo por lo que luchar: 404 bytes en Python 2.7
undergroundmonorail
¿Su primer ejemplo de salida no contiene uno más -de lo que debería?
Ventero
@Ventro ¡Estás en lo correcto! Sabía cómo escribiría un programa para hacer esto, pero en realidad no lo hice hasta ahora, así que mis salidas de muestra fueron a mano. ¡Uy! (También arruiné la segunda salida de muestra, pero la arreglé antes de que hubiera respuestas.)
undergroundmonorail

Respuestas:

9

GolfScript, 128 caracteres

~](:k/.{{1,%{,}%.!,+}%}:^~{' ':s*}%.{,}%$-1=:9{s*\+9~)>'|'n}+%\zip^.{,~}%$0=){.~[s]*@+>{s\+-2>}%}+%zip{9)s*\n}%\[9s*'+''--'k*n]\

La entrada debe proporcionarse en STDIN como números separados por espacios.

Puedes probar el ejemplo aquí .

Código comentado:

# Parse the input into an 2D array of digits. The width is saved to variable k
~](:k/

# Apply the code block ^ to a copy of this array
.
{                # begin on code block
  {              # for each line
   1,%           #   split at 0s (1, => [0]) (leading, trailing, multiple 0s are 
                 #   removed because of operator % instead of /)
   {,}%          #   calculate the length of each run of 1s                 
   .!,+          #   special case: only zeros, i.e. []
                 #   in this case the ! operator yiels 1, thus [0], else []
  }%             # end for
}:^              # end of code block
~                # apply

# Format row headers
{' ':s*}%        # join numbers with spaces
.{,}%$-1=:9      # calulate the length of the longest row header
                 # and save it to variable <9>
{                # for each row
  s*\+           #   prepend padding spaces
  9~)>           #   and cut at length <9> from the right
  '|'n           #   append '|' and newline
}+%              # end for

# Format column headers
\zip^            # transpose input array and apply the code block ^
                 # i.e. calculate length of runs
.{,~}%$0=)       # determine (negative) length of the longest column header
{                # for each column
  .~[s]*@+       #   prepend enough spaces
  >              #   and cut at common length (from right)
  {s\+-2>}%      #   format each number/empty to 2 chars width
}+%              # end for
zip              # transpose column header into output lines
{9)s*\n}%        # prepend spaces to each line and append newline

# Inject separator line
\[
9s*              # spaces
'+'              # a plus sign
'--'k*           # k (width) number of '--'
n                # newline
]\
Howard
fuente
1
+1 bien, aprendí bastantes buenos trucos de esta publicación
Cristian Lupascu
Me las arreglé para jugar golf en 123 caracteres: ~](:k/.zip\]{{1,%{,}%.!,+}%}/{' ':^*}%{.{,}%$-1=}:f~:r{^*\+r~)>'|'n}+%\f{.~)\[^]*@+>{^\+-2>}%}+%zip{r)^*\n}%r^*'+''--'k*n]((por alguna razón lettercount.com dice 125 caracteres si lo copia, pero le aseguro que son 123 caracteres). Se han cambiado algunas partes del algoritmo, pero la mayoría sigue siendo la misma. También cambié algunos nombres de variables (tener 9 como variable es inteligente, pero también confuso), pero puede volver a cambiarlos si lo desea.
Volatilidad
7

Ruby, 216 255

n=$*.shift.to_i;k=*$*.each_slice(n)
u=->k{k.map{|i|r=i.join.scan(/1+/).map{|i|"%2d"%i.size}
[*["  "]*n,*r[0]?r:" 0"][-n,n]}}
puts u[k.transpose].transpose.map{|i|" "*(n-~n)+i*""},"  "*n+?++"--"*n,u[k].map{|i|i*""+?|}

Si bien esto no produce la salida de muestra exacta dada en la pregunta, sigue las especificaciones. La única diferencia con los ejemplos es que imprimo algunos espacios iniciales / líneas nuevas.

Ejemplo:

$ ruby nonogram.rb 15 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 1 0 0 0 0 0 1 1 1 1 0 0 0 0 1 1 1 0 0 1 0 1 0 0 1 0 0 0 1 1 1 1 1 0 1 0 1 1 1 1 0 0 0 1 1 1 1 1 0 1 1 1 1 0 0 0 0 0 1 1 1 1 1 0 0 0 1 1 0 0 0 0 0 0 1 1 1 0 0 0 0 1 1 0 0 0 1 0 0 0 1 0 0 0 1 0 1 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1
# empty lines removed for brevity
                                  1                       1  
                                1 1 3       3 5   5 3   3 1  
                                7 2 3 2 4 2 3 210 2 3 0 4 215
                              +------------------------------
                             2|
                             1|
                             1|
                             1|
                             1|
                           1 1|
                       3 3 1 1|
                       1 5 1 1|
                         3 5 3|
                         1 5 1|
                         1 3 1|
                     1 1 1 1 1|
               1 1 1 1 1 1 1 1|
                          11 3|
                          11 3|

Registro de cambios:

  • 240 -> 231: Se cambió el formato de entrada para usar argumentos de línea de comando en lugar de stdin.
  • 231 -> 230: Eliminó un espacio moviendo la verificación de valor de chunka map.
  • 230 -> 226: Restar [nil] lugar de llamar Array#compact.
  • 226 -> 216: Simplifica la generación de pistas.
Ventero
fuente
Imprime algunas líneas y espacios adicionales, pero hasta ahora en todas mis pruebas han cumplido con la especificación "0 o más", por lo que es bueno. Sin embargo, ahora te advierto que si empiezo a ver números flotando en el aire a la izquierda de mi monitor, tendré que descalificar esta respuesta :)
undergroundmonorail
1
@undergroundmonorail: La salida se imprime en una forma tal que length(leading spaces + numbers to the left) == 2*ne height(leading newlines + numbers at the top) == n... así que mientras el monitor es lo suficientemente grande para 3*n+1 × 2*n+2los personajes, que no debería tener que descalificar a mí. :)
Ventero
4

Rubí, 434

n=$*[i=0].to_i
a,b=[],[]
a.push $*.slice!(1..n)*""while $*.size>1
(b.push a.map{|c|c[i]}*"";i+=1)while i<n
a,b=*[a,b].map{|c|c.map{|d|e=d.split(/[0]+/).map(&:size).select{|f|f>i=0}.map &:to_s;(e.size>0)?e:[?0]}}
m,k=*[a,b].map{|c|c.map(&:size).max}
s="  "*m
k.times{$><<s+"  "+b.map{|c|(" "+((c.size==k-i)?(c.shift):(" ")))[-2..-1]}*"";i+=1;puts}
puts s+" "+?++?-*n*2
a.each{|c|puts"  "*(m-c.size)+" "+c.map{|d|(" "+d)[-2..-1]}*""+?|}
afuo
fuente
¿Cómo ejecutas esto? Lo intenté ruby $yourprogram <<< $inputpero lo conseguí ruby_nanograms:7:in '<main>': undefined method '+' for nil:NilClass (NoMethodError).
undergroundmonorail
@undergroundmonorail ruby nonogram.rb 2 1 0 0 1para un 2x2
afuo el
Esta es una buena respuesta, pero no imprime 0la cuarta y última columna en el segundo ejemplo.
undergroundmonorail
Me acabo de dar cuenta de que +------... la línea está sangrada por demasiados espacios también.
undergroundmonorail
1
@undergroundmonorail Se corrigieron ambos.
afuous
4

GolfScript 149 147

El código

~](:s/.zip{{[0]%{,`}%['0']or}%.{,}%$)\;:¶;{.,¶\-[' ']*\+}%}:f~¶:v;\[f~]\zip{{{.,2\-' '*\+}%''*}:d2*)' '*:z\+{puts}:o~}%z(;'+'s'-'2**++o~{d'|'+o}/

Ediciones:

  • espacio inútil eliminado
  • definió una función reutilizable de un carácter para putsguardar un carácter más

Demostraciones en línea

Una versión algo anotada del código

# split lines
~](:s/

# make transposed copy
.zip

#prepare numbers to show in the header
{{[0]%{,`}%['0']or}%.{,}%$)\;:¶;{.,¶\-[' ']*\+}%}:f~¶:v;

# prepare numbers to show in the left column
\[f~]\zip

#print header (vertical hints)
{  {{.,2\-' '*\+}%''*}:d~  2*)' '*:z\+puts}%

#print first line
z(;'+'s'-'2**++puts

#print horizontal hints
~{d'|'+ puts}/
Cristian Lupascu
fuente
4

Javascript (E6) 314 334 357 410

N=(d,l)=>{J=a=>a.join(''),M=s=>(s.match(/1+/g)||['']).map(x=>x.length),f=' '.repeat(d+1),c=[n='\n'],o=n+f+'+'+'--'.repeat(d);for(i=-1;++i<d;)c[i]=M(J(l.map((e,p)=>p%d-i?'':e))),o+=n+(f+J(M(J(l).substr(i*d,d)).map(P=n=>n>9?n:n<10?' '+n:'  '))+'|').slice(-d-2);for(;--i;)o=n+f+' '+J(c.map(e=>P(e.pop())))+o;return o}

Sin golf

N=(d,l)=> {
  J = a => a.join(''),
  M = s => (s.match(/1+/g)||['']).map(x=>x.length),
  f=' '.repeat(d+1), c=[n='\n'], o=n+f+'+'+'--'.repeat(d);
  for(i = -1; ++i < d;)
    c[i] = M(J(l.map((e,p)=>p%d-i?'':e))),
    o += n+(f+J(M(J(l).substr(i*d,d)).map(P=n=>n>9?n:n<10?' '+n:'  '))+'|').slice(-d-2);
  for(;--i;)
    o=n+f+' '+J(c.map(e=>P(e.pop())))+o;
  return o
}

Uso

N(5,[0,1,1,1,0,1,1,0,1,1,1,0,1,0,1,1,1,0,1,1,0,1,1,1,0])

N(15,[0,0,0,0,0,0,0,0,0,0,0,0,0,1,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,0,0,0,0,0,0,0,0,1,0,0,0,0,0,1,1,1,1,0,0,0,0,1,1,1,0,0,1,0,1,0,0,1,0,0,0,1,1,1,1,1,0,1,0,1,1,1,1,0,0,0,1,1,1,1,1,0,1,1,1,1,0,0,0,0,0,1,1,1,1,1,0,0,0,1,1,0,0,0,0,0,0,1,1,1,0,0,0,0,1,1,0,0,0,1,0,0,0,1,0,0,0,1,0,1,1,0,1,0,1,0,1,0,1,0,1,0,1,0,1,1,1,1,1,1,1,1,1,1,1,1,0,1,1,1,1,1,1,1,1,1,1,1,1,1,1,0,1,1,1])

Editar historial

1 Se eliminó la expresión regular utilizada para encontrar columnas
. Overkill 2 Simpler es mejor. Salida a una cadena, no a una matriz. Función auxiliar eliminada FILL (F)
3 Aún más simple. No puedo hacerlo mejor que esto. Todavía no se puede comparar con Golfscript :(

edc65
fuente
Agradable. También probé una versión de Javascript, pero terminé con algo alrededor de 500 bytes y decidí que era demasiado grande para ponerlo aquí. Sería bueno publicar una versión no protegida con nombres de variables originales (si aún la tiene). Además, ¿cómo ejecutas esto? Si copio y lo pego en la ventana de la consola de Chrome, obtengo "ReferenceError: Invalid left-hand left in task". ¿Hay algo que cambiar o agregar antes de ejecutar?
tigrou
@tigrou lo siento, el "=>" sintax solo funciona en firefox. Variables: c sugerencias de columnas, d dimensión, l lista de entrada, o salida, i variable de bucle, q y z temp
edc65
-15 bytes :)
nderscore
@nderscore jugando con el código, obtuve 326. En su código R no se inicializa (error fácil cuando lo intenta una y otra vez ...)
edc65
1

R, 384 caracteres

a=scan();p=function(x)paste(x,collapse="");P=paste0;s=sapply;l=length;f=function(i)lapply(apply(matrix(a[-1],nr=a,b=T),i,rle),function(x)if(any(x$v)){x$l[!!x$v]}else{0});g=function(j,i)apply(s(j,function(x)sprintf("%2s",c(rep("",max(s(j,l))-l(x)),x))),i,p);c=P(g(f(1),2),"|");d=g(f(2),1);h=p(rep(" ",nchar(c[1])-1));e=P(h,"+",p(rep("-",nchar(d[1]))));d=P(h," ",d);cat(d,e,c,sep="\n")

Con muescas y algunas explicaciones:

a=scan() #Takes input

p=function(x)paste(x,collapse="") #Creates shortcuts
P=paste0
s=sapply
l=length

#This function finds the number of subsequent ones in a line (using rle = run length encoding).
#It takes 1 or 2 as argument (1 being row-wise, 2 column-wise
f=function(i)lapply(apply(matrix(a[-1],nr=a,b=T),i,rle),function(x)if(any(x$v)){x$l[!!x$v]}else{0})

#This function takes the result of the previous and format the strings correctly (depending if they are rows or columns)
g=function(j,i)apply(s(j,function(x)sprintf("%2s",c(rep("",max(s(j,l))-l(x)),x))),i,p)

c=paste0(g(f(1),2),"|") #Computes the rows
d=g(f(2),1) #Computes the columns
h=p(rep(" ",nchar(c[1])-1)) 
e=paste0(h,"+",p(rep("-",nchar(d[1])))) #Prepare vertical border
d=paste0(h," ",d) #Pad column indices with spaces
cat(d,e,c,sep="\n") #Prints

Uso:

> a=scan();p=function(x)paste(x,collapse="");P=paste0;s=sapply;l=length;f=function(i)lapply(apply(matrix(a[-1],nr=a,b=T),i,rle),function(x)if(any(x$v)){x$l[!!x$v]}else{0});g=function(j,i)apply(s(j,function(x)sprintf("%2s",c(rep("",max(s(j,l))-l(x)),x))),i,p);c=P(g(f(1),2),"|");d=g(f(2),1);h=p(rep(" ",nchar(c[1])-1));e=P(h,"+",p(rep("-",nchar(d[1]))));d=P(h," ",d);cat(d,e,c,sep="\n")
1: 15 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 1 0 0 0 0 0 1 1 1 1 0 0 0 0 1 1 1 0 0 1 0 1 0 0 1 0 0 0 1 1 1 1 1 0 1 0 1 1 1 1 0 0 0 1 1 1 1 1 0 1 1 1 1 0 0 0 0 0 1 1 1 1 1 0 0 0 1 1 0 0 0 0 0 0 1 1 1 0 0 0 0 1 1 0 0 0 1 0 0 0 1 0 0 0 1 0 1 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1 1 1 1 1 1 1 1 1 1 1 1 0 1 1 1
227: 
Read 226 items
                    1                       1  
                  1 1 3       3 5   5 3   3 1  
                  7 2 3 2 4 2 3 210 2 3 0 4 215
                +------------------------------
               2|
               1|
               1|
               1|
               1|
             1 1|
         3 3 1 1|
         1 5 1 1|
           3 5 3|
           1 5 1|
           1 3 1|
       1 1 1 1 1|
 1 1 1 1 1 1 1 1|
            11 3|
            11 3|
plannapus
fuente
1

C - 511

C definitivamente no fue hecho para formatear bien la salida. El recuento de caracteres incluye solo espacios necesarios / nuevas líneas.

La entrada es de STDIN, números separados por espacios.

#define P printf
#define L(x) for(x=0;x<s/2+1;x++)
#define l(x) for(x=0;x<s;x++)
#define B(x,y) x[i][j]||y==s/2?P("%2d",x[i][j]):P("  ");
a[50][50],i,j,s,h[25][50],v[50][25],H[50],V[50],x[25],y[25];
main(){
    scanf("%d",&s);
    L(j)x[j]=y[j]=s/2+1;
    l(i)l(j)scanf("%d",&a[i][j]);
    for(i=s-1;i>=0;i--)
        for(j=s-1;j>=0;j--)
            a[i][j]?
                !H[j]&&(x[j]--,H[j]=1),
                h[x[j]][j]++,
                !V[i]&&(y[i]--,V[i]=1),
                v[i][y[i]]++:
            (H[j]=V[i]=0);
    L(i){
        L(j)P("  ");
        P(" ");
        l(j)B(h,i);
        P("\n");
    }
    L(i)P("  ");
    P("+");
    l(i)P("--");
    P("\n");
    l(i){
        L(j)B(v,j);
        P("|\n");
    }
}
Allbeert
fuente
1

Han pasado unos días y nadie respondió en Python, así que aquí está mi intento (probablemente bastante pobre):

Python 2.7 - 404 397 380 bytes

def p(n,m):
 k=str.join;l=[];g=lambda y:[['  ']*(max(map(len,y))-len(t))+t for t in[[' '*(a<10)+`a`for a in map(len,k("",c).split('0'))if a]or[' 0']for c in y]]
 while m:l+=[map(str,m[:n])];m=m[n:]
 x=g(l);j=k('\n',['  '*max(map(len,x))+'+'+k("",a)for a in zip(*[list(a)+['--']for a in g(zip(*l))])]);return j.replace('+',' ',j.count('+')-1)+'\n'+k('\n',[k("",a+['|'])for a in x])

Pronto publicaré una versión sin golf, pero por el momento creo que es bastante legible. :)

EDITAR: Mientras escribía la versión sin golf, noté algunas mejoras que podría hacer que sumaran bastante! Por alguna razón que no puedo explicar, ahora tiene nuevas líneas adicionales en la parte superior y espacios a la izquierda (aunque no creo que haya cambiado nada funcional), pero aún cumple con las especificaciones.¡Ya viene la versión sin golf!

Sin golf:

def nonogram(board_size, pixels):
    def hints(board):
        output = []
        for row in board:
            # Convert the row to a string of 1s and 0s, then get a list of strings
            # that came between two 0s.
            s = "".join(row).split('0')

            # A list of the length of each string in that list.
            l = map(len, s)

            # We now have our horizontal hints for the board, except that anywhere
            # there were two 0s consecutively we have a useless 0.
            # We can get rid of the 0s easily, but if there were no 1s in the row at
            # all we want exactly one 0.
            # Solution:
            output.append([h for h in l if h != 0] or [0])
            # In this context, `foo or bar` means `foo if foo is a truthy value, bar
            # otherwise`.
            # An empty list is falsey, so if we strip out all the strings we hardcode
            # the 0.
        return output

    def num_format(hints):
        # For both horizontal and vertical hints, we want a space before single-
        # digit numbers and no space otherwise. Convert hints to strings and add
        # spaces as necessary.
        output = []

        for row in hints:
            output.append([' '*(a < 10) + str(a) for a in row])
            # Multiplying a string by n repeats it n times, e.g. 'abc'*3=='abcabcabc'
            # The only numbers that need a space are the ones less than 10.
            # ' '*(a < 10) first evaluates a < 10 to get a True or False value.
            # Python automatically converts True to 1 and False to 0.
            # So, if a is a one digit number, we do `' '*(1) + str(a)`.
            # If it's a two digit number, we do `' '*(0) + str(a)`.
        return output

    def padding(hints):
        output = []
        longest = max(map(len, hints)) # how long is the longest row?
        for row in hints:
            output.append(['  ']*(longest - len(row)) + row)
            # Add '  ' to the beginning of every row until it's the same length
            # as the longest one. Because all hints are two characters wide, this
            # ensures all rows of hints are the same length.
        return output

    board = []

    while pixels: # non-empty list == True
        # Make a list of the first (board_size) pixels converted to strings, then
        # add that list to board. Remove those pixels from the list of pixels.
        # When pixels is empty, board has a seperate list for each row.
        board.append([str(n) for n in pixels[:board_size]])
        pixels = pixels[board_size:]

    horizontal_hints = padding(num_format(hints(board)))

    vertical_hints = padding(num_format(hints(zip(*board))))
    # zip(*l) is a common way to transpose l.
    # zip([1,2,3], [4,5,6], [7,8,9]) == [(1, 4, 7), (2, 5, 8), (3, 6, 9)]
    # the star operator unpacks an iterable so the contents can be used as
    # multiple arguments, so
    # zip(*[[1,2,3],[4,5,6],[7,8,9]]) is the same as what we just did.
    # Transposing the board and getting the horizontal hints gives us the
    # vertical hints of the original, but transposed. We'll transpose it back,
    # but we'll also add '--' to the end of all of them to make up the line
    vertical_hints = zip(*[a + ['--'] for a in vertical_hints])

    # add n spaces, where n is the length of the longest horizontal hint, plus
    # one space to the beginning of each line in the vertical hints, then join
    # with newlines to make it all one string.
    vertical_hints = '\n'.join(['  '*max(map(len, horizontal_hints)) + '+' +
                               ''.join(a) for a in vertical_hints])

    # find the number of plus signs in the string
    # replace the first (that many - 1) plus signs with spaces
    vertical_hints = vertical_hints.replace('+', ' ', vertical_hints.count('+')-1)

    # add a pipe to each row of horizontal hints, then join it with newlines
    horizontal_hints = '\n'.join([''.join(a + ['|']) for a in horizontal_hints])

    # add and return
    return vertical_hints + '\n' + horizontal_hints

Se hicieron algunos cambios por razones de legibilidad ( gdivididos en tres funciones con nombre, listas complejas de comprensión hechas en forbucles), pero lógicamente funciona exactamente de la misma manera.

Es por eso que es confuso que este no imprima espacios adicionales y líneas nuevas, mientras que el golf lo hace. ¯ \ _ (ツ) _ / ¯

metro subterráneo
fuente
1
Umm, no puedo encontrar tu solución. (lo siento, solo una broma terrible con respecto al recuento de personajes, no te preocupes por mí :))
Pomo de la puerta
@dor Aha! ¡Intenta hacer chistes de código de error HTTP ahora! : P
undergroundmonorail