From f6c80ebcfaaee0c8d3509cbf8758e37051f09751 Mon Sep 17 00:00:00 2001 From: Reza Behzadan Date: Mon, 1 Jun 2020 04:31:15 +0430 Subject: [PATCH] add solver --- sudoku/sudoku.lua | 159 ++++++++++++++------------------------------ ui/sudokucanvas.lua | 3 + 2 files changed, 53 insertions(+), 109 deletions(-) diff --git a/sudoku/sudoku.lua b/sudoku/sudoku.lua index fcbdb85..040b7e1 100644 --- a/sudoku/sudoku.lua +++ b/sudoku/sudoku.lua @@ -94,6 +94,15 @@ function showBoard(board) 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) local row = 0 local column = 0 @@ -433,129 +442,60 @@ function checkSearchSpace(board, searchSpace) return true end -function backTrace2(board, s) - local s = s or buildSearchSpace(board) - local x, y = findFirstEmptyCell(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) - if r then return r end - end - end -end - -function backTrace3(board, tried) - -- dbg() - local tried = tried or createEmptyBoard() - 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 +function findFirstLeastOption(searchSpace) + local min = 9 + local x, y + for i = 1, 9 do + for j = 1, 9 do + local len = #searchSpace[i][j] + if len > 0 and len < min then + min = len + x = i + y = j end end end + return x, y, min 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 +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 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 +function rollBack(trace) + assert(#trace > 1, "Something is very wrong!") + table.remove(trace) end function solve(board) - -- allPairs(board) - applyFillSinglesRepeatedly(board) - local ok, finished = checkBoard(board) - 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 + local t = {} + t[1] = {board=cloneBoard(board),} + return backTrace(t) 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 { loadBoard = loadBoard, cloneBoard = cloneBoard, @@ -566,4 +506,5 @@ return { showBoard = showBoard, checkBoardsEqual = checkBoardsEqual, saveBoard = saveBoard, + solve = solve, } diff --git a/ui/sudokucanvas.lua b/ui/sudokucanvas.lua index 4dbdf88..8248afd 100644 --- a/ui/sudokucanvas.lua +++ b/ui/sudokucanvas.lua @@ -433,6 +433,9 @@ function SudokuCanvas:keypressed(key) undo(self) elseif key == "f" then 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 self.cursor.x = x self.cursor.y = y