- xml - AJAX/Jquery XML 解析
- 具有多重继承的 XML 模式
- .net - 枚举序列化 Json 与 XML
- XML 简单类型、简单内容、复杂类型、复杂内容
这个周末我研究了一个基于回溯算法的数独求解器 (Ruby quiz)。数独被加载到一个包含 81 个整数(9x9 网格)的数组 sudoku_arr
中,其中 0 是空位。有一个 valid?
方法检查 sudoku_arr
是否可以是一个有效的数独。
official backtracking algorithm是这样的:在下一个空点尝试值,检查它是否有效数独,如果不是将值增加 1(最多 9),如果有效继续并在下一个点尝试第一个值,如果不是增加前一个 0 的值。
因此我们必须跟踪之前的数组,这是我出错的地方,我不确定是否可以解决。下面我的代码中不起作用的部分是 SudokuSolver
类中的 solve_by_increasing_value_previous_index
。这是代码:
require 'pp'
sudoku_str = "
+-------+-------+-------+
| _ 6 _ | 1 _ 4 | _ 5 _ |
| _ _ 8 | 3 _ 5 | 6 _ _ |
| 2 _ _ | _ _ _ | _ _ 1 |
+-------+-------+-------+
| 8 _ _ | 4 _ 7 | _ _ 6 |
| _ _ 6 | _ _ _ | 3 _ _ |
| 7 _ _ | 9 _ 1 | _ _ 4 |
+-------+-------+-------+
| 5 _ _ | _ _ _ | _ _ 2 |
| _ _ 7 | 2 _ 6 | 9 _ _ |
| _ 4 _ | 5 _ 8 | _ 7 _ |
+-------+-------+-------+"
class SudokuException < StandardError
attr_reader :sudoku_arr
def initialize(message, sudoku_arr)
super(message)
@sudoku_arr = sudoku_arr
end
end
class Sudoku
attr_accessor :sudoku_arr,
:index_of_tried_value,
:tried_value
def initialize(sudoku_arr, index_of_tried_value, tried_value)
@sudoku_arr = sudoku_arr
@index_of_tried_value = index_of_tried_value
@tried_value = tried_value
end
def rows_valid?
rows_have_unique_values?(@sudoku_arr)
end
def columns_valid?
rows_have_unique_values?(@sudoku_arr.each_slice(9).to_a.transpose.flatten!)
end
def squares_valid?
tmp_a = @sudoku_arr.each_slice(3).to_a
rows_have_unique_values?(
(tmp_a[0] << tmp_a[3] << tmp_a[6] << tmp_a[1] << tmp_a[4] << tmp_a[7] <<
tmp_a[2] << tmp_a[5] << tmp_a[8] << tmp_a[9] << tmp_a[12] << tmp_a[15] <<
tmp_a[10] << tmp_a[13] << tmp_a[16] << tmp_a[11] << tmp_a[14] << tmp_a[17] <<
tmp_a[18] << tmp_a[21] << tmp_a[24] << tmp_a[19] << tmp_a[22] << tmp_a[25] <<
tmp_a[20] << tmp_a[23] << tmp_a[26]).flatten!)
end
def valid?
rows_valid? && columns_valid? && squares_valid?
end
def rows_have_unique_values?(arr)
(arr[0,9]- [0]).uniq.size == (arr[0,9]- [0]).size &&
(arr[9,9]- [0]).uniq.size == (arr[9,9]- [0]).size &&
(arr[18,9]-[0]).uniq.size == (arr[18,9]-[0]).size &&
(arr[27,9]-[0]).uniq.size == (arr[27,9]-[0]).size &&
(arr[36,9]-[0]).uniq.size == (arr[36,9]-[0]).size &&
(arr[45,9]-[0]).uniq.size == (arr[45,9]-[0]).size &&
(arr[54,9]-[0]).uniq.size == (arr[54,9]-[0]).size &&
(arr[63,9]-[0]).uniq.size == (arr[63,9]-[0]).size &&
(arr[72,9]-[0]).uniq.size == (arr[72,9]-[0]).size
end
end
class SudokuSolver
attr_accessor :sudoku_arr,
:indeces_of_zeroes
def initialize(str)
@sudoku_arr = str.gsub(/[|\+\-\s]/,"").gsub(/_/,'0').split(//).map(&:to_i)
@indeces_of_zeroes = []
@sudoku_arr.each_with_index { |e,index| @indeces_of_zeroes << index if e.zero? }
end
def solve
sudoku_arr = @sudoku_arr
try_index = @indeces_of_zeroes[0]
try_value = 1
sudoku = Sudoku.new(sudoku_arr, try_index, try_value)
solve_by_increasing_value(sudoku)
end
def solve_by_increasing_value(sudoku)
if sudoku.tried_value < 10
sudoku.sudoku_arr[sudoku.index_of_tried_value] = sudoku.tried_value
if sudoku.valid?
pp "increasing_index..."
solve_by_increasing_index(sudoku)
else
pp "increasing_value..."
sudoku.tried_value += 1
solve_by_increasing_value(sudoku)
end
else
pp "Increasing previous index..."
solve_by_increasing_value_previous_index(sudoku)
end
end
def solve_by_increasing_index(sudoku)
if sudoku.sudoku_arr.index(0).nil?
raise SudokuException(sudoku.sudoku_arr.each_slice(9)), "Sudoku is solved."
end
sudoku.index_of_tried_value = sudoku.sudoku_arr.index(0)
sudoku.tried_value = 1
solve_by_increasing_value(sudoku)
end
def solve_by_increasing_value_previous_index(sudoku)
# Find tried index and get the one before it
tried_index = sudoku.index_of_tried_value
previous_index = indeces_of_zeroes[indeces_of_zeroes.index(tried_index)-1]
# Setup previous Sudoku so we can go back further if necessary:
# Set tried index back to zero
sudoku.sudoku_arr[tried_index] = 0
# Set previous index
sudoku.index_of_tried_value = previous_index
# Set previous value at index
sudoku.tried_value = sudoku.sudoku_arr[previous_index]
pp previous_index
pp sudoku.tried_value
# TODO Throw exception if we go too far back (i.e., before first index) since Sudoku is unsolvable
# Continue business as usual by increasing the value of the previous index
solve_by_increasing_value(sudoku)
end
end
sudoku_solver = SudokuSolver.new(sudoku_str)
sudoku_solver.solve
不幸的是,代码没有回溯到开头。代码打印:
"increasing_index..." "increasing_value..." "increasing_value..." "increasing_value..." "increasing_value..." "increasing_value..." "increasing_value..." "increasing_value..." "increasing_value..." "Increasing previous index..." 16 2
在一个循环中,直到它抛出一个 SystemStackError
,因为堆栈级别太深。
发生的情况是“回溯”不超过一个索引。当 solve_by_increasing_value_previous_index
转到前一个索引时,它会在那里取前一个值。在这种情况下,它是 2,但 2 不起作用,因此我们应该将它减少到 1 并继续,如果这不起作用,则丢弃 2 并再次将其设为 0,然后返回更远的地方。
不幸的是,我没有看到实现此算法的简单方法。 (我想到的是一个 @too_much_looping
变量,它在 solve_by_increasing_value_previous_index
被调用时递增,并在 81 次后重置。但这只会帮助进行再返回一次,我们就不能循环回到开头。\
希望有人能给我一些帮助!也非常欢迎一般代码评论,我怀疑这不是 100% 地道的 Ruby。
最佳答案
我还没有仔细阅读您的代码,但回溯算法归结为以下内容:
int solve(char board[81]) {
int i, c;
if (!is_valid(board)) return 0;
c = first_empty_cell(board);
if (c<0) return 1; /* board is full */
for (i=1; i<=9; i++) {
board[c] = i;
if (solve(board)) return 1;
}
board[c] = 0;
return 0;
}
这是一个递归函数,在它找到的第一个空单元格中尝试从 1
到 9
的每个值(由 first_empty_cell()
返回) ).如果这些值都没有导致解决方案,那么您一定位于搜索树的死分支上,因此可以将有问题的单元格重置为零(或您用来指示未填充单元格的任何值)。
当然,还有很多其他事情可以让您的软件更快地得出解决方案,但就回溯而言,仅此而已。
好的,我现在正在浏览您的代码。看起来您正在将 index_of_tried_value
存储为数独网格的属性。那行不通的。您需要将此值存储在求解器例程的局部变量中,以便在您回溯搜索树时将其压入堆栈并恢复。
关于ruby - 如何使具有回溯算法的数独求解器返回?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25108258/
我的填充数独板的代码如下所示: public class SudokuBoard { static int N = 9; static int[][] grid = new int[N
我正在使用“强力”随机方法创建一个数独生成器。我已经能够使用以下代码检查 x/y 轴是否有重复数字: for(l=0; l<9; l++){//Makes all vertical work.
我有一个数独谜题求解器,需要我们使用递归。问题是我检查可用空间的 boolean 值应该通过引用更新当前位置,但事实并非如此。什么会导致这个? public boolean solve() {
我正在尝试为我的项目制作一个数独游戏,但如果我增加数独Grid中的空白空间数量,代码只会抛出异常arrayoutofbounds但是无法弄清楚它来自哪里。 k 是网格中空白空间的数量。 我没有尝试过任
当我尝试编写一个解决数独问题的程序时,我的代码中有一个错误。 我的程序可以运行,但运行效果不佳。它只解决程序中的第一行。 我认为我在那行代码上犯了错误: int ft_rezolva(int **t)
这是我的。该方法应输出该空间中所有可用数字的数组。出于某种原因,这不会过滤掉相同的框/行/列。应该如何正确编码? public int[] getAllowedValues(int row, in
我遇到了一个问题。我是 Java 的新手,正在尝试尝试比以前更复杂的东西。这是我自己的个人文件输入和主要方法与其他方法的一些蚕食源的组合。我对递归仍然很生疏。出于某种原因,更改二维数组“板”中值的分配
嘿,我无法让代码比较给定行或列和 block 的整数,以确保这些参数中没有重复项。我不知道用 3 种不同的方法分离这三个约束是否是一个好主意,或者只是尝试一次完成所有的操作。 public stati
我正在研究 javascript 数独游戏,就输入验证而言,我很困惑。当每个值都输入到数独表中时,我有一个 onkeyup 触发一个函数,我试图用它来确保没有值进入已经存在这样一个值的行/列/框。这是
我正在研究一个小的个人数独游戏并试图扩展它。 到目前为止,我使用递归回溯方法使“求解”部分正常工作,该方法在设法求解递归时返回 true。 现在我正在尝试构建一个独特的解决方案板生成器,并且我在网上找
数独回溯法 int xx = (pos.getX() / 3) * 3; int yy = (pos.getY() / 3) * 3; for (int y =
我是 c++ 的新手,在做作业(数独)时遇到了问题。 说明说:“你必须创建一个新板作为当前板的拷贝(使用复制构造函数并使用 new) 从堆中分配板。” 我试过了(写在board.cc中): #incl
我目前正在使用 java 制作数独游戏,但我似乎无法弄清楚如何正确地循环我的扫描仪。到目前为止,它打印了这个: . 1 . | 3 . . | 8 . . 5 . 9 | 6 . . | 7 . .
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 12 个月前关闭。 Improve
我正在一个由随机生成的数字(从 1 到 4)填充的多维数组中创建一个 4x4 数独游戏。现在我必须搜索重复项并用其他随机数替换它们,直到单行、单列和四个 2x2 子矩阵上都有唯一的数字。我怎样才能做到
我正在制作一个数独板 GUI,它应该看起来像这样 http://www.sudoku.4thewww.com/Grids/grid.jpg 由于某种原因,它只显示最后一个 3*3 板。如果有人能告诉我
嘿,我写了这个程序来解决数独问题,但它只适用于数独矩阵的几个单元格,而其他单元格则返回 0 。你能明白这有什么问题吗?我是 Java 编码新手,无法编写简单的程序真的很痛苦。 public class
这里是 Stack Overflow 的第一个定时器! 首先,我正在使用 JavaFX 创建一个数独求解器。我一切正常,但是,我遇到的唯一问题是创建粗体 3x3 大块,每个大块内有 3x3 单元格。我
我的逻辑求解算法有问题。它很好地解决了具有大量提示的谜题,它只是解决了少于 45 条线索的谜题。 这是求解的算法。 Immutable 是一个 boolean 值,用于确定该值是否可以更改。 cell
我正在尝试使用 Python 线性优化库 Pulp 来解决 killer 数独问题。 https://en.wikipedia.org/wiki/Killer_sudoku 这是我迄今为止的尝试,添加
我是一名优秀的程序员,十分优秀!