@@ -10,6 +10,7 @@ class BoggleSolver
10
10
public $ board = array ();
11
11
public $ boardLookup = array ();
12
12
13
+ public $ lastDictTime ;
13
14
public $ lastSolveTime ;
14
15
15
16
// for now, 3 - 11 letter words only
@@ -22,8 +23,12 @@ class BoggleSolver
22
23
'n ' , 'ne ' , 'e ' , 'se ' , 's ' , 'sw ' , 'w ' , 'nw ' ,
23
24
);
24
25
25
- public function __construct ()
26
+ // $dictFile should be relative to this file
27
+ public function __construct ($ dictFile = "" )
26
28
{
29
+ if ($ dictFile ) {
30
+ static ::$ dictFile = $ dictFile ;
31
+ }
27
32
}
28
33
29
34
public function getWords ()
@@ -33,11 +38,13 @@ public function getWords()
33
38
34
39
public function loadDict ()
35
40
{
41
+ $ start = microtime (true );
42
+
36
43
$ this ->dict = array ();
37
44
$ words = $ this ->getWords ();
38
45
39
- // Create a boggle friendly lookup table. This is basically a half-assed
40
- // search tree so we know when we can stop checking for words on the
46
+ // Create a boggle friendly lookup table. This is basically a half-assed
47
+ // search tree so we know when we can stop checking for words on the
41
48
// current path.
42
49
foreach ($ words as $ word ) {
43
50
@@ -46,26 +53,24 @@ public function loadDict()
46
53
continue ;
47
54
}
48
55
56
+ $ wordLen = strlen ($ word );
57
+
49
58
// max length is mostly here for memory reasons
50
- if (strlen ( $ word ) < static ::$ minLen ) {
59
+ if ($ wordLen < static ::$ minLen ) {
51
60
continue ;
52
61
}
53
- if (strlen ($ word ) > static ::$ maxLen || strlen ($ word ) > $ this ->size * $ this ->size ) {
62
+ if ($ wordLen > static ::$ maxLen ) {
63
+ continue ;
64
+ }
65
+ if ($ wordLen > $ this ->size * $ this ->size ) {
54
66
continue ;
55
67
}
56
68
57
69
$ ptr = &$ this ->dict ;
58
70
59
- for ($ i = 0 ; $ i <= static ::$ maxLen ; $ i ++) {
60
- if ($ i == strlen ($ word )) {
61
- $ ptr [] = $ word ;
62
- continue ;
63
- } else if ($ i > strlen ($ word )) {
64
- continue ;
65
- }
66
-
71
+ for ($ i = 0 ; $ i < $ wordLen ; $ i ++) {
67
72
$ letter = $ word [$ i ];
68
- if ($ letter == 'Q ' && $ word [$ i + 1 ] == 'U ' ) {
73
+ if ($ letter == 'Q ' && isset ( $ words [ $ i + 1 ]) && $ word [$ i + 1 ] == 'U ' ) {
69
74
$ letter = 'Qu ' ;
70
75
$ i ++;
71
76
}
@@ -74,9 +79,12 @@ public function loadDict()
74
79
}
75
80
$ ptr = &$ ptr [$ letter ];
76
81
}
82
+ $ ptr [] = $ word ;
77
83
}
78
84
79
85
unset($ words );
86
+
87
+ $ this ->lastDictTime = microtime (true ) - $ start ;
80
88
}
81
89
82
90
public function loadBoard ($ board )
@@ -157,6 +165,9 @@ public function displayBoard($lineEnd = "\n")
157
165
$ colPtr = $ rowPtr ;
158
166
while ($ colPtr !== null ) {
159
167
$ board .= $ colPtr ->letter ;
168
+ if ($ colPtr ->letter != "Qu " ) {
169
+ $ board .= " " ;
170
+ }
160
171
$ colPtr = &$ colPtr ->e ;
161
172
}
162
173
$ board .= $ lineEnd ;
@@ -181,7 +192,7 @@ public function findWords()
181
192
182
193
$ words = array_merge (
183
194
$ words ,
184
- $ this ->findWordsFromOneTile ($ ptr , null )
195
+ $ this ->findWordsFromOneTile ($ ptr , null , $ words )
185
196
);
186
197
187
198
$ ptr ->visited = false ;
@@ -200,10 +211,10 @@ public function findWords()
200
211
201
212
$ this ->lastSolveTime = microtime (true ) - $ start ;
202
213
203
- return array_keys ( $ words) ;
214
+ return $ words ;
204
215
}
205
216
206
- public function findWordsFromOneTile ($ boardPtr , $ dictPtr = null , $ words = array ())
217
+ public function findWordsFromOneTile ($ boardPtr , $ dictPtr = null , & $ words = array (), $ path = array ())
207
218
{
208
219
if ($ dictPtr === null ) {
209
220
$ dictPtr = &$ this ->dict ;
@@ -215,10 +226,15 @@ public function findWordsFromOneTile($boardPtr, $dictPtr = null, $words = array(
215
226
return array ();
216
227
}
217
228
229
+ $ path [] = $ boardPtr ->id ;
230
+
218
231
$ dictPtr = &$ dictPtr [$ curLetter ];
219
232
220
233
if (isset ($ dictPtr [0 ])) {
221
- $ words [$ dictPtr [0 ]] = 1 ;
234
+ if (!isset ($ words [$ dictPtr [0 ]])) {
235
+ $ words [$ dictPtr [0 ]] = array ();
236
+ }
237
+ $ words [$ dictPtr [0 ]][] = $ path ;
222
238
}
223
239
224
240
foreach (static ::$ dirs as $ dir ) {
@@ -229,12 +245,12 @@ public function findWordsFromOneTile($boardPtr, $dictPtr = null, $words = array(
229
245
continue ;
230
246
}
231
247
232
- // if we're going diagonal, make sure the two adjacent tiles
248
+ // if we're going diagonal, make sure the two adjacent tiles
233
249
// haven't already been pathed to each other
234
250
if (strlen ($ dir ) == 2 ) {
235
- $ d1 = $ dir[ 0 ] ;
236
- $ d2 = $ dir[ 1 ] ;
237
- if ($ boardPtr ->$ d1 ->pathTo == $ boardPtr ->$ d2 ->id ||
251
+ $ d1 = substr ( $ dir, 0 , 1 ) ;
252
+ $ d2 = substr ( $ dir, 1 , 1 ) ;
253
+ if ($ boardPtr ->$ d1 ->pathTo == $ boardPtr ->$ d2 ->id ||
238
254
$ boardPtr ->$ d2 ->pathTo == $ boardPtr ->$ d1 ->id
239
255
) {
240
256
continue ;
@@ -246,7 +262,7 @@ public function findWordsFromOneTile($boardPtr, $dictPtr = null, $words = array(
246
262
$ boardPtr ->$ dir ->pathTo = $ boardPtr ->id ;
247
263
$ boardPtr ->pathTo = $ boardPtr ->$ dir ->id ;
248
264
249
- $ newWords = $ this ->findWordsFromOneTile ($ boardPtr ->$ dir , $ dictPtr , $ words );
265
+ $ newWords = $ this ->findWordsFromOneTile ($ boardPtr ->$ dir , $ dictPtr , $ words, $ path );
250
266
$ words = array_merge ($ words , $ newWords );
251
267
252
268
// unset those flags since we're picking a new path after this
0 commit comments