问题描述
我正在尝试学习反向跟踪,虽然我想将所有解决方案添加到我的数组列表中,但它只包含我的方法找到的第一个解决方案。我检查了 isSafe 方法,它是正确的。唯一的问题是我的 QueensList 方法。你能解释一下如何将所有解决方案添加到我的数组列表中吗? 例如,对于 4x4 表,我的 ArrayList 的元素是
[0,1,0]
[0,1]
[1,0]
public static boolean queensList(int[][] aList,int r,int k,ArrayList<int[][]> qL) {
if (r >= k) {
qL.add(aList);
return true;
}
for (int i = 0; i < k; i++) {
if (isSafe(aList,r,i)) {
aList[r][i] = 1;
if (queensList(aList,r + 1,k,qL))
return true;
aList[r][i] = 0;
}
}
return false;
}
public static boolean isSafe(int[][] trying,int c) {
for (int i = 0; i < trying.length; i++) {
if (trying[r][i] == 1 || trying[i][c] == 1)
return false;
}
int i = 0;
while (r - i < trying.length && 0 <= r - i && c - i < trying.length && 0 <= c - i || r - i < trying.length && 0 <= r - i && c + i < trying.length && 0 <= c + i
|| r + i < trying.length && 0 <= r + i && c - i < trying.length && 0 <= c - i || r + i < trying.length && 0 <= r + i && c + i < trying.length && 0 <= c + i) {
if (r - i < trying.length && 0 <= r - i && c - i < trying.length && 0 <= c - i && trying[r - i][c - i] == 1)
return false;
if (r - i < trying.length && 0 <= r - i && c + i < trying.length && 0 <= c + i && trying[r - i][c + i] == 1)
return false;
if (r + i < trying.length && 0 <= r + i && c - i < trying.length && 0 <= c - i && trying[r + i][c - i] == 1)
return false;
if (r + i < trying.length && 0 <= r + i && c + i < trying.length && 0 <= c + i && trying[r + i][c + i] == 1)
return false;
i++;
}
return true;
}
解决方法
你的函数 queensList
的退出条件应该是只有当 (r >= k)
时,这意味着当 r
行结束 k
时,女王仍然是安全的,这是一个有效的解决方案。
但是在您的实现中,当您在 for 循环中找到解决方案时,您也返回了,这意味着它没有完成整个 for
循环以找到所有可能的解决方案。
所以对你的回溯函数做一个小的改变,就像这样:
public static void queensList(int[][] aList,int r,int k,ArrayList<int[][]> qL) {
//exit condition
if (r >= k) {
qL.add(aList);
return;
}
for (int i = 0; i < k; i++) {
if (isSafe(aList,r,i)) {
aList[r][i] = 1;
queensList(aList,r + 1,k,qL);
aList[r][i] = 0;
}
}
}