Crea un puzzle Nonogram


24

Un nonogramma è un puzzle logico bidimensionale che assomiglia a questo (schermate del gioco Pixelo , il mio gioco nonogramma preferito):

Una scheda nonogramma vuota

L'obiettivo del gioco è capire quale immagine codificano quei numeri. Le regole sono semplici: un numero su una colonna o riga significa che da qualche parte in quella colonna o riga, che molte caselle sono riempite in una riga. Ad esempio, la riga inferiore nell'immagine sopra non deve contenere caselle, mentre quella sopra deve avere tutte le caselle riempite. La terza fila dal basso ha 8 caselle piene e saranno tutte in fila.

Due o più numeri per la stessa colonna o riga indicano che ci sono più "esecuzioni" di caselle piene, con almeno uno spazio tra, con quelle lunghezze. L'ordine è conservato. Ad esempio, ci sono tre caselle riempite nella colonna a destra dell'immagine sopra, almeno uno spazio sotto di esse, e poi un'altra casella riempita.

Ecco lo stesso puzzle, quasi completato:

Una scheda nonogramma quasi finita

(Le X non sono importanti, sono solo un suggerimento che il giocatore lascia a se stesso per dire "Questo quadrato non è sicuramente riempito". Pensa alle bandiere in Minesweeper. Non hanno significato delle regole.)

Spero che tu possa vedere che, per esempio, le colonne centrali con suggerimenti che dicono "2 2" hanno due serie di 2 scatole piene.

La tua missione, se scegli di accettarlo, è quella di scrivere un programma o una funzione che creerà un puzzle come questo. Ti viene data la dimensione della tavola come un singolo intero (5 <= n <= 50) su stdin o come argomento (non c'è motivo per cui un puzzle nonogramma debba essere quadrato, ma per questa sfida sarà). Successivamente, ti verrà data una serie di 1 e 0 che rappresentano i quadrati pieni e non riempiti nell'immagine, rispettivamente. Le prime n sono la riga superiore, quindi la riga successiva, ecc. Verrà restituito o stampato per stdout una scheda di celle 2 * 1 (perché hanno un aspetto migliore e ti dà spazio per i suggerimenti di 2 cifre per una colonna ), tutti vuoti, con suggerimenti corrispondenti ai dati di input.

Formato di output

Formato di output

Campione

Ingresso:

./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])

Immagine:

Immagine di primo esempio

Produzione:

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

Ingresso:

./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'

Immagine:

Immagine del secondo esempio

Produzione:

                   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|

chiarimenti

  • Il tuo output non deve necessariamente essere un puzzle risolvibile. Non tutti i nonogrammi sono risolvibili, ma non è questo il tuo problema. Basta emettere i suggerimenti che corrispondono all'input, che si tratti o meno di un buon puzzle.
  • È consentito un programma che accetta argomenti sulla riga di comando. Questo è un po 'come detto sopra, ma è possibile avere un'idea sbagliata. Ecco a cosa servono i chiarimenti.
  • La stampa 0di una riga o colonna senza caselle compilate è obbligatoria. Non lo dico con parole da nessuna parte ma è nei dati di esempio.

Ho quasi finito con la mia soluzione. Possiamo supporre che non ci sarà un numero di 3 cifre di caselle in una riga o colonna?
afuoso

2
@voidpigeon: 5<=n<=50è la specifica, quindi non possono esserci numeri a 3 cifre
Kyle Kanos,

Dopo aver pubblicato questa domanda, ho iniziato a lavorare su una soluzione da solo. Non lo posterò ancora (come da questa meta risposta), ma posterò il mio conteggio dei byte in modo che voi ragazzi abbiate qualcosa per cui lottare: 404 byte in Python 2.7
undergroundmonorail

Il tuo primo esempio di output non ne contiene uno in più -di quanto dovrebbe?
Ventero,

@Ventro Hai ragione! Sapevo come avrei scritto un programma per farlo, ma in realtà non l'ho fatto fino ad ora, quindi i miei output di esempio erano fatti a mano. Oops! (Ho anche rovinato il secondo output del campione, ma l'ho corretto prima che ci fossero delle risposte.)
undergroundmonorail

Risposte:


9

GolfScript, 128 caratteri

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

L'input deve essere fornito su STDIN come numeri separati da spazio.

Puoi testare l'esempio qui .

Codice commentato:

# 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
]\

1
+1 bello, ho imparato parecchi trucchi da questo post
Cristian Lupascu il

Sono riuscito a golf con 123 caratteri: ~](:k/.zip\]{{1,%{,}%.!,+}%}/{' ':^*}%{.{,}%$-1=}:f~:r{^*\+r~)>'|'n}+%\f{.~)\[^]*@+>{^\+-2>}%}+%zip{r)^*\n}%r^*'+''--'k*n]((per qualche motivo lettercount.com dice 125 caratteri se lo copi, ma ti assicuro, sono 123 caratteri). Alcune parti dell'algoritmo sono state modificate, ma la maggioranza è sempre la stessa. Ho anche cambiato alcuni nomi di variabili (avere 9 come variabile è intelligente, ma anche confuso), ma puoi cambiarli se vuoi.
Volatilità il

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*""+?|}

Sebbene ciò non produca l'esatto esempio di output fornito nella domanda, segue le specifiche. L'unica differenza con gli esempi è che stampo alcuni spazi / nuove righe iniziali.

Esempio:

$ 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|

changelog:

  • 240 -> 231: formato di input modificato per utilizzare gli argomenti della riga di comando anziché lo stdin.
  • 231 -> 230: eliminato uno spazio spostando il controllo del valore da chunka map.
  • 230 -> 226: Sottrai [nil]invece di chiamare Array#compact.
  • 226 -> 216: semplifica la generazione di suggerimenti.

Stampi qualche nuova riga e spazio extra, ma finora in tutti i miei test hanno adattato la specifica "0 o più", quindi stai bene. Ti avverto ora, tuttavia, se comincio a vedere i numeri fluttuare nell'aria a sinistra del mio monitor, dovrò squalificare questa risposta :)
undergroundmonorail

1
@undergroundmonorail: l'output è stampato in modo tale che length(leading spaces + numbers to the left) == 2*ne height(leading newlines + numbers at the top) == n... finché il monitor è abbastanza grande per i 3*n+1 × 2*n+2personaggi, non dovresti squalificarmi. :)
Ventero,

4

Rubino, 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]}*""+?|}

Come si esegue questo? Ho provato ruby $yourprogram <<< $inputma ho ottenuto ruby_nanograms:7:in '<main>': undefined method '+' for nil:NilClass (NoMethodError).
undergroundmonorail,

@undergroundmonorail ruby nonogram.rb 2 1 0 0 1for a 2x2
afuous

Questa è una buona risposta, ma non si stampa 0per la quarta ultima colonna nel secondo esempio.
undergroundmonorail,

Ho appena notato che il +------ linea ... è rientrata da uno di troppi spazi.
undergroundmonorail,

1
@undergroundmonorail Risolti entrambi.
afuoso

4

GolfScript 149 147

Il codice

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

modifiche:

  • rimosso spazio inutile
  • definita una funzione di un carattere riutilizzabile per putssalvare un altro carattere

Demo online

Una versione in qualche modo annotata del codice

# 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}/

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}

Ungolfed

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])

Modifica cronologia

1 Regexp rimosso usato per trovare le colonne. Overkill
2 Più semplice è meglio. Output su una stringa, non su un array. Funzione helper rimossa FILL (F)
3 Ancora più semplice. Non posso fare di meglio. Non è ancora possibile confrontare con Golfscript :(


Bello. Ho provato anche una versione Javascript ma ho terminato con qualcosa di circa 500 byte e ho deciso che ero troppo grande per metterlo qui. Sarebbe bene pubblicare una versione non golfata con nomi di variabili originali (se ce l'hai ancora). Inoltre, come si esegue questo? Se copio incollalo nella finestra della console di Chrome ottengo "ReferenceError: lato sinistro non valido nell'assegnazione". C'è qualcosa da cambiare o aggiungere prima di correre?
Tigrou,

@tigrou scusa la sintassi "=>" funziona solo su Firefox. Variabili: c colunns hints, d dimension, l list of input, o output, i loop variabile, q and z temp
edc65


@nderscore armeggiando con il codice, ho 326. Nel tuo codice R non è inizializzato (errore facile quando riprovi ancora e ancora ...)
edc65

1

R, 384 caratteri

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 rientri e alcune spiegazioni:

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|

1

C - 511

C non è stato creato per la formattazione corretta dell'output. Il conteggio dei caratteri include solo gli spazi necessari / nuove linee.

L'input proviene da STDIN, numeri separati da spazi.

#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");
    }
}

1

Sono passati alcuni giorni e nessuno ha risposto in Python, quindi ecco il mio (probabilmente piuttosto scarso) tentativo:

Python 2.7 - 404 397 380 byte

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])

Pubblicherò presto una versione non giocata, ma al momento penso che sia abbastanza leggibile. :)

EDIT: Mentre scrivevo la versione non golfata, ho notato alcuni miglioramenti che potrei fare per renderlo abbastanza significativo! Per qualche motivo che non posso spiegare, ora ha nuove righe in alto e spazi a sinistra (anche se non credo di aver cambiato nulla di funzionale), ma soddisfa ancora le specifiche.La versione Ungolfed sta arrivando!

Ungolfed:

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

Sono state apportate alcune modifiche per motivi di leggibilità ( gsuddivise in tre funzioni denominate, complesse comprensioni di elenchi trasformate in forloop) ma logicamente funziona esattamente allo stesso modo.

Questo è il motivo per cui è confuso che questo non stampi spazi extra e newline, mentre quello golfizzato. ¯ \ _ (ツ) _ / ¯


1
Umm, non riesco a trovare la tua soluzione. (scusa, solo una terribile battuta sul conteggio dei personaggi, non preoccuparti di me :))
Maniglia della porta

@dor Aha! Prova ora a fare battute sul codice di errore HTTP! : P
undergroundmonorail
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.