Rubinowy, 272 bajtów
Dwie niepotrzebne nowe znaki są dodawane do kodu po obu stronach funkcji zagnieżdżonej, gaby poprawić czytelność. Są one wyłączone z wyniku. Znaki f=przypisujące funkcję anonimową do zmiennej są również wykluczone.
Format wyjściowy to 0lub 1na pytanie zamiast natywnego truei Ruby false. Nowa linia (zamiast spacji) służy do oddzielenia wartości logicznej i słowa. Rozumiem, że jest to akceptowalna interpretacja wymagań wyjściowych, ale jeśli nie, wpływ na liczbę bajtów byłby niewielki.
f=->n,l{c=''
x=[p=6*n,1,-p,-1]
(m=3*p*n).times{|i|c<<(5+i/n%6-i/n/p&6==6?65+rand(26):i%p==p-1?10:46)}
q=m+3*n
puts c
g=->w,i,d{w==''?$r=1:c[i]<?A?g[w,(i+x[d])%q,d^1]:w[0]==c[i]&&4.times{|j|g[w[1..-1],(i+x[j])%q,j^1]}}
l.each{|w|$r=0
m.times{|i|c[i]>?@&&g[w,i,0]}
puts $r,w}}
Wynik
Po około 50 takich połączeniach:
f[4,['PPCG','CODE','GOLF','ANT','CAN','CUBE','WORD','WALK','SPELL']]
W końcu otrzymałem następujący wynik z 2 trafieniami. ANTjest w prawym dolnym rogu i idzie w górę, i ANjest udostępniany przez CAN, z Crundy owijania do góry po lewej stronie.
....KCAAXRHT...........
....ALRZXRKL...........
....NDDLCMCT...........
....ETQZHXQF...........
........FYYUSRZX.......
........CFNPAUVX.......
........ZTJVHZVQ.......
........AUWKGVMC.......
............XWKSDWVZ...
............DPLUVTZF...
............DMFJINRJ...
............ZRXJIAFT...
0
PPCG
0
CODE
0
GOLF
1
ANT
1
CAN
0
CUBE
0
WORD
0
WALK
0
SPELL
Wyjaśnienie
Konkretne rozwinięcie wybranego sześcianu zostało wybrane częściowo ze względu na łatwość rysowania, ale głównie ze względu na łatwość wyszukiwania.
Znaki inne niż alfabet (kropki plus znak nowej linii na końcu każdej linii) są ważną częścią pola, na którym mrówka może chodzić.
Wyszukiwanie jest wykonywane przez funkcję rekurencyjną g, która jest zagnieżdżona w funkcji f. Jeśli przekazane słowo jest pustym ciągiem, wyszukiwanie jest zakończone i $rustawione na 1. Jeśli mrówka znajduje się na kwadracie, który odpowiada pierwszej literze słowa, wyszukiwanie jest kontynuowane we wszystkich czterech kierunkach: funkcja jest wywoływana ponownie ze słowem skróconym przez usunięcie pierwszej litery. W takim przypadku parametr kierunku jest ignorowany. Przenoszenie odbywa się poprzez rekurencyjne wywoływanie z indeksem komórki zmienionym o wartości w x.Wynik dodawania jest brany modulo do wielkości siatki plus dodatkowa połowa linii. Oznacza to, że dolna linia zawija się do góry i odwrotnie, z prawidłowym przesunięciem w poziomie.
Jeśli mrówka znajduje się na kwadracie innym niż litera, musi poruszać się zygzakiem po schodach, aż znajdzie kwadrat z literą. Będzie zygzakiem w kierunku południowo-wschodnim lub północno-zachodnim. Jest to symulowane przez wywołania rekurencyjne z dparametrem XORed z 1 za każdym razem, aby śledzić jej ruch. Dopóki nie dojdzie do kwadratu z następną literą, słowo wejściowe nie ulega skróceniu. Dogodnie można tego dokonać za pomocą tej samej rekurencji, co w przypadku wyszukiwania w obszarze z literami. Różnica polega na tym, że rekursja ma tylko jedną gałąź, gdy mrówka znajduje się w obszarze białych znaków, a nie 4 w obszarze litery.
Skomentowany kod
->n,l{ #n=square size, l=list of words to search
c='' #empty grid
x=[p=6*n,1,-p,-1] #offsets for south, east, north, west. p is also number of characters per line
(m=3*p*n).times{|i| #m=total cells in grid. for each cell
c<<(5+i/n%6-i/n/p&6==6? #apppend to c (according to the formula)
65+rand(26): #either a random letter
i%p==p-1?10:46) #or a "whitespace character" (newline, ASCII 10 or period, ASCII 46)
}
q=m+3*n #offset for vertical wraparound = grid size plus half a row.
puts c #print grid
g=->w,i,d{ #search function. w=word to search for, i=start index in grid, d=direction
w==''? #if length zero, already found,
$r=1: #so set flag to 1. Else
c[i]<?A? #if grid cell is not a letter
g[w,(i+x[d])%q,d^1]: #recursively call from the cell in front, with the direction reflected in NW-SE axis
w[0]==c[i]&& #else if the letter on the grid cell matches the start of the word
4.times{|j| #for each direction (iterate 4 times, each time a different direction is "in front")
g[w[1..-1],(i+x[j])%q,j^1]} #recursively call from the cell in front. Chop first letter off word.
} #Direction parameter is XORed (reflected in NW-SE axis) in case ant hits whitespace and has to zigzag.
l.each{|w| #for each word in the list
$r=0 #set global variable $r to zero to act as a flag
m.times{|i|c[i]>?@&&g[w,i,0]} #call g from all cells in the grid that contain a letter
puts $r,w} #output flag value and word
}