This is a simple project about making an automatic Sudoku solver. It can be run as a CLI tool, by running:
dotnet run --project SudokuSolver --configuration Release -- --board {BOARD} --solver {SOLVER} --solution {SOLUTIONFILE}
Where:
BOARD
has to be a string of rows in the sudoku (total of 81 numbers), where 0 represents a blank spaceSOLVER
is the solver you want. The CLI tool will tell you what options are availableSOLUTION
is optional. It describes the file you want the solved board to be outputted to. If none is given, no file will be made.
This project is also available as a dotnet tool on the NuGet Package Manager.
Then you can install it by writing dotnet tool install SudokuSolver
and use it as follows:
sudokusolver --board {BOARD} --solver {SOLVER} --solution {SOLUTIONFILE}
Where the arguments means the same as above.
There are four core solvers in this projects, three brute force (B) and one logical (L). There are also presets that combine some of these solvers, to simplify the job for the following solver.
This solver is a sort of a "baseline" solver. It is pretty bad, since it selects what cell to try and fill out in random order. This leads to a lot of weird and bad branches that simply requires more time and a lot more calls.
Sequential backtrack selects cells sequentially, column by row. It can also be visualised in the following example, where the sequence is from A until the end of the board (thanks to SudokuWiki.org for the image)
Where it will try and assign all legal values to the A cell first, then to the B cell, etc. If a cell becomes impossible to assign to, we backtrack to the previous cell.
This is a backtrack solver that selects cells based on the cardinality of possibilities for rows. It sorts its possible cells to select first by fewest possibilities for the entire row and then sort it by the cell in that row that have the fewest assignment possibilities itself. As an example, one can look at the following image (the small numbers mean the numbers that can legally be assigned to that cell):
It can be seen that it is row F that have the fewest total possible assignments, so that row takes priority. When it comes to selecting a cell in that row, we take the cell in the row that have the fewest possibilities too. So in this case we will try F1 and F6 before we try F9.
This is perhaps the most interesting solver of them all. It attempts to solve the board by means of logical deduction. The set of logical strategies that this solver currently implements are:
- Last Remaining Cell in Box/Row/Column
- Naked Pairs
- Naked Tripples
- Hidden Pairs
- Hidden Tripples
- Pointing Pairs/Tripples
- Box Line Reduction
- X-Wings
- Singles Chain
Many of the boards cant be solved by logical deduction (at least with the current strategies) but paired with any of the backtrack solvers makes a powerful solver.
Benchmark is run on 1971 different Sudoku boards with a 2s time limit. The results are ordered by solved instances, then by lowest average search time and finally by average calls.
Solver | Solved | Avg Search (ms) | Avg Calls | Max Search (ms) | Min Search (ms) | Max Calls | Min Calls |
---|---|---|---|---|---|---|---|
LogicalWithCardinalityBacktrack | 1971 | 2.65 | 28287.67 | 67.45 | 0.13 | 1156304 | 2 |
LogicalWithSequentialBacktrack | 1971 | 3.67 | 43340.56 | 143.25 | 0.13 | 1903421 | 2 |
CardinalityBacktrack | 1970 | 11.49 | 203634.19 | 1284.8 | 0.03 | 32545851 | 42 |
SequentialBacktrack | 1967 | 22.08 | 424353.67 | 1888.97 | 0.02 | 34359357 | 43 |
LogicalWithRandomBacktrack | 1376 | 55.64 | 14892574.54 | 2001.76 | 0.13 | 82421544 | 2 |
Logical | 1043 | 0.58 | 3.23 | 4.58 | 0.14 | 19 | 1 |
RandomBacktrack | 367 | 420.29 | 38631112.87 | 1972.91 | 0.03 | 57641799 | 176 |
I would like to give a special thanks to Andrew Stuart, the owner of the SudokuWiki.org website, for giving a bunch of great description of the different strategies that are involved in solving sudokus!