add solver

This commit is contained in:
Reza Behzadan 2020-06-01 04:31:15 +04:30
parent dccf9b3616
commit f6c80ebcfa
2 changed files with 53 additions and 109 deletions

View File

@ -94,6 +94,15 @@ function showBoard(board)
end end
end end
function checkBoardsEqual(b1, b2)
for i = 1, 9 do
for j = 1, 9 do
if b1[i][j] ~= b2[i][j] then return false end
end
end
return true
end
function iterZones(board) function iterZones(board)
local row = 0 local row = 0
local column = 0 local column = 0
@ -433,128 +442,59 @@ function checkSearchSpace(board, searchSpace)
return true return true
end end
function backTrace2(board, s) function findFirstLeastOption(searchSpace)
local s = s or buildSearchSpace(board) local min = 9
local x, y = findFirstEmptyCell(board) local x, y
for _, v in ipairs(s[x][y]) do for i = 1, 9 do
local b = cloneBoard(board) for j = 1, 9 do
b[x][y] = v local len = #searchSpace[i][j]
allPairs(b) if len > 0 and len < min then
local ok, finished = checkBoard(b) min = len
if finished then return b end x = i
if ok and checkSearchSpace(b) then y = j
local r = backTrace(b)
if r then return r end
end end
end end
end end
return x, y, min
end
function backTrace(trace)
while true do
local currentStep = trace[#trace]
applyFillSinglesRepeatedly(currentStep.board)
local ok, finished = checkBoard(currentStep.board)
if not ok then rollBack(trace); goto endOfTheLoop end
if finished then return currentStep.board end
local s = buildSearchSpace(currentStep.board)
ok = checkSearchSpace(currentStep.board, s)
if not ok then rollBack(trace); goto endOfTheLoop end
if not currentStep.pointer then
local x, y = findFirstLeastOption(s)
currentStep.options = s[x][y]
currentStep.pointer = 0
currentStep.x = x
currentStep.y = y
end
currentStep.pointer = currentStep.pointer + 1
if currentStep.pointer > #currentStep.options then rollBack(trace); goto endOfTheLoop end
local nextStep = {}
nextStep.board = cloneBoard(currentStep.board)
nextStep.board[currentStep.x][currentStep.y] = currentStep.options[currentStep.pointer]
trace[#trace+1] = nextStep
::endOfTheLoop::
end
end
function backTrace3(board, tried) function rollBack(trace)
-- dbg() assert(#trace > 1, "Something is very wrong!")
local tried = tried or createEmptyBoard() table.remove(trace)
for x = 1, 9 do
for y = 1, 9 do
if tried[x][y] == 0 and board[x][y] == 0 then
tried[x][y] = 1
print(x, y)
local s = buildSearchSpace(board)
for _, v in ipairs(s[x][y]) do
local b = cloneBoard(board)
b[x][y] = v
allPairs(b)
local ok, finished = checkBoard(b)
if finished then return b end
if ok and checkSearchSpace(b) then
local r = backTrace(b, tried)
if r then return r end
end
end
end
end
end
end
function backTrace4(board, cx, cy)
local s = buildSearchSpace(board)
for x = 1, 9 do
for y = 1, 9 do
if x ~= cx and y ~=cy then
for _, v in ipairs(s[x][y]) do
local b= cloneBoard(board)
b[x][y] = v
print(x, y, v)
-- allPairs(b)
applyFillSinglesRepeatedly(b)
local ok, finished = checkBoard(b)
if finished then return b end
if ok and checkSearchSpace(b) then
local r = backTrace(b, x, y)
if r then return r end
end
end
end
end
end
end
function backTrace(board)
local s = buildSearchSpace(board)
for x = 1, 9 do
for y = 1, 9 do
for _, v in ipairs(s[x][y]) do
local b= cloneBoard(board)
b[x][y] = v
-- allPairs(b)
applyFillSinglesRepeatedly(b)
local ok, finished = checkBoard(b)
print(x, y, v, ok, finished)
showBoard(b)
io.read(1)
if finished then return b end
if ok and checkSearchSpace(b) then
local r = backTrace(b, x, y)
if r then return r end
end
end
return
end
end
end end
function solve(board) function solve(board)
-- allPairs(board) local t = {}
applyFillSinglesRepeatedly(board) t[1] = {board=cloneBoard(board),}
local ok, finished = checkBoard(board) return backTrace(t)
if not ok then print("something's wrong!"); return end
if finished then showBoard(board); return end
local r = backTrace(board)
if r then
showBoard(r)
else
print "Could not solve!"
end end
end
function sHelper(n)
local fn = string.format("/home/reza/p/lua/22.sudoku/boards/%03d.txt", n)
local board = loadBoard(fn)[1]
solve(board)
end
function checkBoardsEqual(b1, b2)
for i = 1, 9 do
for j = 1, 9 do
if b1[i][j] ~= b2[i][j] then return false end
end
end
return true
end
-- b = loadBoard("b.txt")[1]
-- applyFillSinglesRepeatedly(b)
-- s = buildSearchSpace(b)
-- allPairs(b)
return { return {
loadBoard = loadBoard, loadBoard = loadBoard,
@ -566,4 +506,5 @@ return {
showBoard = showBoard, showBoard = showBoard,
checkBoardsEqual = checkBoardsEqual, checkBoardsEqual = checkBoardsEqual,
saveBoard = saveBoard, saveBoard = saveBoard,
solve = solve,
} }

View File

@ -433,6 +433,9 @@ function SudokuCanvas:keypressed(key)
undo(self) undo(self)
elseif key == "f" then elseif key == "f" then
sudoku.saveBoard("lastBoard.txt", self.board[1], self.board[2]) sudoku.saveBoard("lastBoard.txt", self.board[1], self.board[2])
elseif key == "c" then
self.board[2] = sudoku.solve(self.board[1])
self:checkBoard()
end end
self.cursor.x = x self.cursor.x = x
self.cursor.y = y self.cursor.y = y