JavaScript (ES6), 595 628 680
Bearbeiten Einige Bereinigung und Zusammenführung:
- Funktion P fusioniert innerhalb Funktion R
- Calc x und z in der gleichen .Map
- wenn Lösung gefunden, Set x auf 0 äußere Schleife verlassen
- und fusionierten definiton Anruf von W
Edit2 mehr Golf, zufällige Füllung verkürzt, äußere Schleife überarbeitet ... siehe Verlauf für etwas lesbarer
Im Gegensatz zu der akzeptierten Antwort, dies für die meisten Eingaben funktionieren. Vermeiden Sie einfach einzelne Buchstaben. Wenn eine Ausgabe gefunden wird, ist sie optimal und verwendet alle drei Richtungen.
Die Einschränkung, das Wiederholen von Wörtern zu vermeiden, ist sehr schwierig. Ich musste bei jedem Schritt, bei dem ich ein Wort zum Gitter hinzufügte, und bei jedem zufälligen Füllzeichen nach wiederholten Wörtern suchen.
Hauptunterfunktionen:
P (w) wahr, wenn das Wort palindrom ist. Ein Palindrom-Wort wird zweimal gefunden, wenn nach wiederholten Wörtern gesucht wird.
R (s) prüft die Wiederholung von Wörtern in Gitter s
Q (s) füllen das Gitter s mit zufälligen Zeichen - es ist rekursiv und rückgängig zu machen, wenn sich ein Wort wiederholt - und können fehlschlagen.
W () rekursiv, versuchen Sie, wenn möglich, ein Raster mit der angegebenen Größe zu füllen.
Die Hauptfunktion verwendet W (), um ein Ausgabegitter zu finden. Dabei wird versucht, das längste Wort in der Eingabe bis zur Summe der Länge aller Wörter zu finden.
F=l=>{
for(z=Math.max(...l.map(w=>(w=w.length,x+=w,w),x=0));
++z<=x;
(W=(k,s,m,w=l[k])=>w?s.some((a,p)=>!!a&&
D.some((d,j,_,r=[...s],q=p-d)=>
[...w].every(c=>r[q+=d]==c?c:r[q]==1?r[q]=c:0)
&&R(r)&&W(k+1,r,m|1<<(j/2))
)
)
:m>12&&Q(s)&&(console.log(''+s),z=x)
)(0,[...Array(z*z-z)+99].map((c,i)=>i%z?1:'\n'))
)
D=[~z,-~z,1-z,z-1,z,-z,1,-1]
,R=u=>!l.some(w=>u.map((a,p)=>a==w[0]&&D.map(d=>n+=[...w].every(c=>u[q+=d]==c,q=p-d)),
n=~([...w]+''==[...w].reverse()))&&n>0)
,Q=(u,p=u.indexOf(1),r=[...'ABCDEFGHIJHLMNOPQRSTUVWXYZ'])=>
~p?r.some((v,c)=>(r[u[p]=r[j=0|c+Math.random()*(26-c)],j]=v,R(u)&&Q(u)))||(u[p]=1):1
//,Q=u=>u.map((c,i,u)=>u[i]=c!=1?c:' ') // uncomment to avoid random fill
}
Ungolfed und erklärt (unvollständig, sorry Leute, es ist eine Menge Arbeit)
F=l=>
{
var x, z, s, q, D, R, Q, W;
// length of longest word in z
z = Math.max( ... l.map(w => w.length))
// sum of all words length in x
x = 0;
l.forEach(w => x += w.length);
for(; ++z <= x; ) // test square size from z to x
{
// grid in s[], each row of len z + 1 newline as separator, plus leading and trailing newline
// given z==offset between rows, total length of s is z*(z-1)+1
// gridsize: 2, z:3, s.length: 7
// gridsize: 3, z:4, s.length: 13
// ...
// All empty, nonseparator cells, filled with 1, so
// - valid cells have a truthy value (1 or string)
// - invalid cells have falsy value ('\n' or undefined)
s = Array(z*z-z+1).fill(1)
s = s.map((v,i) => i % z != 0 ? 1 : '\n');
// offset for 8 directions
D = [z+1, -z-1, 1-z, z-1, z, -z, 1, -1]; // 4 diags, then 2 vertical, then 2 horizontal
// Function to check repeating words
R = u => // return true if no repetition
! l.some( w => // for each word (exit early when true)
{
n = -1 -([...w]+''==[...w].reverse()); // counter starts at -1 or -2 if palindrome word
u.forEach( (a, p) => // for each cell if grid
{
if (a == [0]) // do check if cell == first letter of word, else next word
D.forEach( d => // check all directions
n += // word counter
[...w].every( c => // for each char in word, exit early if not equal
u[q += d] == c, // if word char == cell, continue to next cell using current offset
q = p-d // starting position for cell
)
) // end for each direction
} ) // end for each cell
return n > 0 // if n>0 the word was found more than once
} ) // end for each word
// Recursive function to fill empty space with random chars
// each call add a single char
Q =
( u,
p = u.indexOf(1), // position of first remaining empty cell
r = [...'ABCDEFGHIJHLMNOPQRSTUVWXYZ'] // char array to be random shuffled
) => {
if (~p) // proceed if p >= 0
return r.some((v,c)=>(r[u[p]=r[j=0|c+Math.random()*(26-c)],j]=v,R(u)&&Q(u)))||(u[p]=1)
else
return 1; // when p < 0, no more empty cells, return 1 as true
}
// Main working function, recursive fill of grid
W =
( k, // current word position in list
s, // grid
m, // bitmask with all directions used so far (8 H, 4V, 2 or 1 diag)
w = l[k] // get current word
) => {
var res = false
if (w) { // if current word exists
res = s.some((a,p)=>!!a&&
D.some((d,j,_,r=[...s],q=p-d)=>
[...w].every(c=>r[q+=d]==c?c:r[q]==1?r[q]=c:0)
&&R(r)&&W(k+1,r,m|1<<(j/2))
)
)
}
else
{ // word list completed, check additional constraints
if (m > 12 // m == 13, 14 or 15, means all directions used
&& Q(s) ) // try to fill with random, proceed if ok
{ // solution found !!
console.log(''+s) // output grid
z = x // z = x to stop outer loop
res = x//return value non zero to stop recursion
}
}
return res
};
W(0,s)
}
}
Test in der Firefox / FireBug-Konsole
F (['TRAIN', 'CUBE', 'BOX', 'BICYCLE'])
,T,C,B,O,X,B,H,
,H,R,U,H,L,I,H,
,Y,A,A,B,E,C,B,
,D,H,S,I,E,Y,I,
,H,E,R,L,N,C,T,
,G,S,T,Y,F,L,U,
,H,U,Y,F,O,E,H,
nicht gefüllt
,T,C,B,O,X,B, ,
, ,R,U, , ,I, ,
, , ,A,B, ,C, ,
, , , ,I,E,Y, ,
, , , , ,N,C, ,
, , , , , ,L, ,
, , , , , ,E, ,
F)
,T,A,R,C,S,T,H,
,S,R,R,L,U,D,T,
,T,B,A,T,N,B,P,
,O,B,O,I,S,A,E,
,R,B,A,X,N,H,D,
,M,R,M,O,U,T,H,
,B,I,C,Y,C,L,E,
F (['AA', 'AB', 'AC', 'AD', 'AE', 'AF', 'AG'])
,A,U,B,C,
,T,A,E,Z,
,C,D,O,F,
,Q,C,G,A,
F (['AA', 'AB', 'AC', 'AD', 'AE', 'AF'])
Ausgabe nicht gefüllt - @nathan: Jetzt können Sie kein weiteres A x ohne Wiederholungen hinzufügen . Du brauchst ein größeres Gitter.
,A, ,C,
, ,A,F,
,D,E,B,
AC
Beispielsweise würde der in Ihrem Beispiel verbleibende Buchstabe einen anderen ergeben,CAT
wenn dies der Fall istT
.