Skip to content

Commit b36f858

Browse files
committed
Solver now returns paths along with words
* Updated/improved tests and examples * Dictionary loads a lot faster now * Added timing for dictionary
1 parent 53ef643 commit b36f858

File tree

5 files changed

+129
-76
lines changed

5 files changed

+129
-76
lines changed

README.md

+25-3
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
BoggleSolver
22
----
33

4-
This is a PHP class that solves Boggle. I made it for fun.
4+
This is a PHP class that solves Boggle boards. I made it for fun.
55

66
## Installing
77

@@ -31,7 +31,7 @@ try {
3131
"A M T O".
3232
"L N S T".
3333
"L X T G".
34-
"E T A N";
34+
"E T A N"
3535
);
3636
} catch (\BoggleSolver\BoggleException $e) {
3737
die("exiting on error: " . $e->getMessage());
@@ -41,6 +41,27 @@ try {
4141
$words = $boggle->findWords();
4242
````
4343

44+
`BoggleSolver::findWords()` will return an array where every valid word is the key
45+
and the value is one or more arrays describing the path you take on the board to
46+
replicate finding this word.
47+
48+
Example (in formatted JSON, for clarity):
49+
50+
```json
51+
{
52+
"ANTS":[
53+
[0,5,2,6],
54+
[0,5,10,6],
55+
[14,15,10,6]
56+
],
57+
"ANTA":[
58+
[0,5,10,14]
59+
]
60+
}
61+
```
62+
63+
The numbers in the array represent each tile ID.
64+
4465
See the files in the `examples` directory for more info.
4566

4667
## Tests
@@ -59,4 +80,5 @@ will delete the coverage directory.
5980

6081
© Mike Watson
6182

62-
Released under the [MIT license](http://opensource.org/licenses/MIT). See the `LICENSE` file.
83+
Released under the [MIT license](http://opensource.org/licenses/MIT). See the
84+
`LICENSE` file.

examples/example01.php

+4-3
Original file line numberDiff line numberDiff line change
@@ -38,11 +38,12 @@
3838

3939
echo "Found {$numWords} words:\n\n";
4040

41-
foreach ($words as $word) {
42-
echo $word . "\n";
41+
foreach ($words as $word => $paths) {
42+
echo "{$word}\n";
4343
}
4444

4545
echo "\n";
4646

4747
// lastSolveTime will hold the time it took to solve
48-
echo "Solved in {$boggle->lastSolveTime} seconds\n";
48+
echo "Dictionary built in {$boggle->lastDictTime} seconds\n";
49+
echo "Board solved in {$boggle->lastSolveTime} seconds\n";

examples/example02.php

+1-2
Original file line numberDiff line numberDiff line change
@@ -22,13 +22,12 @@
2222

2323
$words = $boggle->findWords();
2424

25-
2625
echo "Boggle Board:\n\n";
2726
echo $boggle->displayBoard();
2827

2928
echo "\n";
3029

31-
foreach ($words as $word) {
30+
foreach ($words as $word => $paths) {
3231
echo $word . "\n";
3332
}
3433

src/BoggleSolver.php

+39-23
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ class BoggleSolver
1010
public $board = array();
1111
public $boardLookup = array();
1212

13+
public $lastDictTime;
1314
public $lastSolveTime;
1415

1516
// for now, 3 - 11 letter words only
@@ -22,8 +23,12 @@ class BoggleSolver
2223
'n', 'ne', 'e', 'se', 's', 'sw', 'w', 'nw',
2324
);
2425

25-
public function __construct()
26+
// $dictFile should be relative to this file
27+
public function __construct($dictFile = "")
2628
{
29+
if ($dictFile) {
30+
static::$dictFile = $dictFile;
31+
}
2732
}
2833

2934
public function getWords()
@@ -33,11 +38,13 @@ public function getWords()
3338

3439
public function loadDict()
3540
{
41+
$start = microtime(true);
42+
3643
$this->dict = array();
3744
$words = $this->getWords();
3845

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
4148
// current path.
4249
foreach ($words as $word) {
4350

@@ -46,26 +53,24 @@ public function loadDict()
4653
continue;
4754
}
4855

56+
$wordLen = strlen($word);
57+
4958
// max length is mostly here for memory reasons
50-
if (strlen($word) < static::$minLen) {
59+
if ($wordLen < static::$minLen) {
5160
continue;
5261
}
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) {
5466
continue;
5567
}
5668

5769
$ptr = &$this->dict;
5870

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++) {
6772
$letter = $word[$i];
68-
if ($letter == 'Q' && $word[$i + 1] == 'U') {
73+
if ($letter == 'Q' && isset($words[$i + 1]) && $word[$i + 1] == 'U') {
6974
$letter = 'Qu';
7075
$i++;
7176
}
@@ -74,9 +79,12 @@ public function loadDict()
7479
}
7580
$ptr = &$ptr[$letter];
7681
}
82+
$ptr[] = $word;
7783
}
7884

7985
unset($words);
86+
87+
$this->lastDictTime = microtime(true) - $start;
8088
}
8189

8290
public function loadBoard($board)
@@ -157,6 +165,9 @@ public function displayBoard($lineEnd = "\n")
157165
$colPtr = $rowPtr;
158166
while ($colPtr !== null) {
159167
$board .= $colPtr->letter;
168+
if ($colPtr->letter != "Qu") {
169+
$board .= " ";
170+
}
160171
$colPtr = &$colPtr->e;
161172
}
162173
$board .= $lineEnd;
@@ -181,7 +192,7 @@ public function findWords()
181192

182193
$words = array_merge(
183194
$words,
184-
$this->findWordsFromOneTile($ptr, null)
195+
$this->findWordsFromOneTile($ptr, null, $words)
185196
);
186197

187198
$ptr->visited = false;
@@ -200,10 +211,10 @@ public function findWords()
200211

201212
$this->lastSolveTime = microtime(true) - $start;
202213

203-
return array_keys($words);
214+
return $words;
204215
}
205216

206-
public function findWordsFromOneTile($boardPtr, $dictPtr = null, $words = array())
217+
public function findWordsFromOneTile($boardPtr, $dictPtr = null, &$words = array(), $path = array())
207218
{
208219
if ($dictPtr === null) {
209220
$dictPtr = &$this->dict;
@@ -215,10 +226,15 @@ public function findWordsFromOneTile($boardPtr, $dictPtr = null, $words = array(
215226
return array();
216227
}
217228

229+
$path[] = $boardPtr->id;
230+
218231
$dictPtr = &$dictPtr[$curLetter];
219232

220233
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;
222238
}
223239

224240
foreach (static::$dirs as $dir) {
@@ -229,12 +245,12 @@ public function findWordsFromOneTile($boardPtr, $dictPtr = null, $words = array(
229245
continue;
230246
}
231247

232-
// if we're going diagonal, make sure the two adjacent tiles
248+
// if we're going diagonal, make sure the two adjacent tiles
233249
// haven't already been pathed to each other
234250
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 ||
238254
$boardPtr->$d2->pathTo == $boardPtr->$d1->id
239255
) {
240256
continue;
@@ -246,7 +262,7 @@ public function findWordsFromOneTile($boardPtr, $dictPtr = null, $words = array(
246262
$boardPtr->$dir->pathTo = $boardPtr->id;
247263
$boardPtr->pathTo = $boardPtr->$dir->id;
248264

249-
$newWords = $this->findWordsFromOneTile($boardPtr->$dir, $dictPtr, $words);
265+
$newWords = $this->findWordsFromOneTile($boardPtr->$dir, $dictPtr, $words, $path);
250266
$words = array_merge($words, $newWords);
251267

252268
// unset those flags since we're picking a new path after this

0 commit comments

Comments
 (0)