Im playing around with a Sudoku solver as shown below. The problem I'm having is that I don't know how to use backtracking to get the solver to go back after it fails with the first try. As shown in the last code snippet the algorithm stops when it hits the first illegal solution and returns Nothing
. How can I make it go back and try another solution until it finds one?
-- Updates a specific sudoku with a value at a specific position
update :: Sudoku -> Pos -> Maybe Int -> Sudoku
-- Returns all the blank possitions in a sudoku
blanks :: Sudoku -> [Pos]
-- checks so that the size is correct 9x9
isSudoku :: Sudoku -> Bool
-- Checks if it is a legal sudoku, no number twise on any line col or box
isOkay :: Sudoku -> Bool
-- Checks if there are no empty cells in the sudoku
isSolved :: Sudoku -> Bool
solve :: Sudoku -> Maybe Sudoku
solve s
| not $ isSudoku s && isOkay s = Nothing
| otherwise = solve' $ pure s
solve' :: Maybe Sudoku -> Maybe Sudoku
solve' Nothing = Nothing --There is no solution
solve' (Just s)
| isSolved s = pure s -- We found a solution
| otherwise = solve' newSud -- Continue looking for solution
where
(p:_) = blanks s
newSud = solveCell (candidates s p)
solveCell [] = Nothing
solveCell (c:cs)
| isOkay $ update s p (pure c) = Just $ update s p (pure c)
| otherwise = solveCell cs
Fails solving and ends up with this as the stopping point.
Just (Sudoku {rows = [
[Just 1,Just 2,Just 3,Just 4,Just 5,Just 6,Just 7,Just 8,Just 9],
[Just 4,Just 5,Just 6,Just 1,Just 2,Just 3,Just 8,Just 7,Nothing]
[Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing],
[Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing],
[Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing],
[Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing],
[Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing],
[Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing],
[Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing,Nothing]]})
I'm going to simplify the problem by writing more generic code. Writing more generic code is frequently easier because there are fewer possibilities.
To search generically we need three things: how to tell when we are
done
with typea -> Bool
, whatbranches
there are to search with typea -> [a]
, and where to start the search from with typea
.Depth-first search generically
The strategy for a depth-first search, which is what we are trying to implement, is simple. If we are
done
, return the result we found. Otherwise find out what branches we can take from here, and try searching each of them in order until one of them returns a result. If there are no branches we can take from here, then we've failed to find a result.A typical implementation of depth-first search, like ours, usually uses the call stack for backtracking. Depth-first search explores all of the possibilities resulting from a decision before exploring other possible decisions. Since it commits to a course of action and either solves the problem or proves that course of action is unsolvable, the state before committing to each course of action can easily be stored on the stack. The stack remembers the state of computations before making a call so that when that call returns that state is restored. This is a perfect match for the states we need to remember for backtracking in depth first search.
The evaluation of
listToMaybe . catMaybes . map go . branches
is driven by lazy evaluation, so the left-most thing is what really always happens first.listToMaybe
is looking for the first solution, trying each possibility fromcatMaybes . map go . branches
in turn until it finds one.catMaybes
is yielding the results frommap go . branches
, throwing out an explored possibility that resulted inNothing
.map go
is making the recursive call for each branch as it is demanded by the other functions.Depth-first search for Sudoku
To use
depthFirstSearch
for your Sudoku problem, we need to provide thedone
andbranches
functions. We already havedone
, it'sisSolved
. We need to provide thebranches
function that finds the legal moves from a position. First we'll find all themoves
.The legal moves are only the ones that are okay.
This is enough to use
depthFirstSearch
Differences from your code
Let's see how
solve'
from above is different from yoursolve'
. They both use the same pieces -isSolved
,isOkay
,blanks
,candidates
, andupdate
but they put them together slightly differently.I'll re-write the
solve'
from above until it looks close to yoursolve'
. First we'll substitute fordepthFirstSearch
and notice thatsolve' = go
and use guards instead ofif ... then ... else
I'll substitute in
legalMoves s
Then substitute for
listToMaybe . catMaybes . map solve'
We could move the
update
intotryInTurn
, but we'd have to keep track ofp
somehow or assume like you did that notisSolved
implies thatblanks
will not be[]
. We'll do the latter, which is what you did.The big difference between this version and your version is that the recursive call to
solve'
happens once for each candidate instead of once for the first okay candidate.Practical concerns
A depth-first sudoku solver is going to have a lot of trouble dealing with the absolutely huge branching factor in sudoku. It might be tenable with the least restrictive move heuristic, which for sudoku would be to choose to make the next move in the position with the fewest okay candidates.