算法学习 八皇后问题的递归实现 java版 回溯思想
2018-02-12 21:20
531 查看
1.问题描述
八皇后问题是一个以国际象棋为背景的问题:如何能够在 8×8 的国际象棋棋盘上放置八个皇后,使得任何一个皇后都无法直接吃掉其他的皇后?为了达到此目的,任两个皇后都不能处于同一条横行、纵行或斜线上。八皇后问题可以推广为更一般的n皇后摆放问题:这时棋盘的大小变为n×n,而皇后个数也变成n。当且仅当 n = 1 或 n ≥ 4 时问题有解。
2.思路分析
回溯法:当把问题分成若干步骤并递归求解时,如果当前步骤没有合法的选择时,则函数即调用上一层的递归,此即为回溯。
在每次的正向递归时是一个试探的过程,将本次的影响带入到下一次的递归过程中去,如果在下一次的函数中,并没有找到一个合适的解去调用在下一层的函数,则证明上一次的函数过程产生了不正确的中间结果,这个结果使得整个函数不能被正确执行了,所以此时应返回对于全局变量的影响(如果产生了如果有必要,回溯法的过程中大多数是有必要返回这个影响的),然后再次去试探。。。一直到达递归的出口。
八皇后问题的思路即:
1):以行的方式去摆放棋子
2):在每次的试探摆放时,检查是否有以前的棋子与本次拟摆放的位置相冲突,(由于是按行摆放,故只需验证是否有同一列和同一对角线即可)
3): 如果产生了冲突,则返回上一层的调用
4):如果产生了符合条件的位置,则进入下一行的判断package lianxi;
import java.util.Scanner;
public class EightQueen {
static int tol = 0,n;
static int C[] ;
static Scanner scan = new Scanner(System.in);
public static void main(String[] args) {
n = scan.nextInt();
C = new int
;
for(int j=0;j<n;j++){
C[j] = 0;
}
search(0);
System.out.println(tol);
}
private static void search(int cur) {
if(cur == n){
tol++;
System.out.println("第"+tol+"种放法:");
for(int i = 0;i< n;i++){
for(int i1 = 0;i1< n;i1++){
if(C[i] == i1)
System.out.print('Q');
else
System.out.print('+');
}
System.out.println();
}
System.out.println();
}else {
for(int j = 0;j<n;j++){
int ok = 1;
C[cur] = j;
for(int k = 0;k<cur ;k++){
if(C[k] == C[cur] || C[k] - k ==C[cur]-cur || C[cur] + cur == C[k] + k ){
八皇后问题是一个以国际象棋为背景的问题:如何能够在 8×8 的国际象棋棋盘上放置八个皇后,使得任何一个皇后都无法直接吃掉其他的皇后?为了达到此目的,任两个皇后都不能处于同一条横行、纵行或斜线上。八皇后问题可以推广为更一般的n皇后摆放问题:这时棋盘的大小变为n×n,而皇后个数也变成n。当且仅当 n = 1 或 n ≥ 4 时问题有解。
2.思路分析
回溯法:当把问题分成若干步骤并递归求解时,如果当前步骤没有合法的选择时,则函数即调用上一层的递归,此即为回溯。
在每次的正向递归时是一个试探的过程,将本次的影响带入到下一次的递归过程中去,如果在下一次的函数中,并没有找到一个合适的解去调用在下一层的函数,则证明上一次的函数过程产生了不正确的中间结果,这个结果使得整个函数不能被正确执行了,所以此时应返回对于全局变量的影响(如果产生了如果有必要,回溯法的过程中大多数是有必要返回这个影响的),然后再次去试探。。。一直到达递归的出口。
八皇后问题的思路即:
1):以行的方式去摆放棋子
2):在每次的试探摆放时,检查是否有以前的棋子与本次拟摆放的位置相冲突,(由于是按行摆放,故只需验证是否有同一列和同一对角线即可)
3): 如果产生了冲突,则返回上一层的调用
4):如果产生了符合条件的位置,则进入下一行的判断package lianxi;
import java.util.Scanner;
public class EightQueen {
static int tol = 0,n;
static int C[] ;
static Scanner scan = new Scanner(System.in);
public static void main(String[] args) {
n = scan.nextInt();
C = new int
;
for(int j=0;j<n;j++){
C[j] = 0;
}
search(0);
System.out.println(tol);
}
private static void search(int cur) {
if(cur == n){
tol++;
System.out.println("第"+tol+"种放法:");
for(int i = 0;i< n;i++){
for(int i1 = 0;i1< n;i1++){
if(C[i] == i1)
System.out.print('Q');
else
System.out.print('+');
}
System.out.println();
}
System.out.println();
}else {
for(int j = 0;j<n;j++){
int ok = 1;
C[cur] = j;
for(int k = 0;k<cur ;k++){
if(C[k] == C[cur] || C[k] - k ==C[cur]-cur || C[cur] + cur == C[k] + k ){
//C[k] - k ==C[cur]-cur C[cur] + cur == C[k] + k分别用于判断是否有与当前位置处于右下对角线和左下对角线的皇后 ok = 0; break; } } if(ok==1)search(cur+1); } } } }
相关文章推荐
- 算法(第四版)学习笔记之二分查找的递归与非递归java实现
- Java基于循环递归回溯实现八皇后问题算法示例
- 算法学习笔记之三:八皇后问题(递归、回溯)
- 经典内部排序算法学习总结(算法思想、可视化、Java代码实现、改进、复杂度分析、稳定性分析)
- 背包算法递归实现,递归转动态规划的一般方法java实现
- 常见算法学习及其Java实现--冒泡算法实现
- 7、JAVA实现快排--分治+递归的思想
- 全序列算法递归实现――回溯
- 算法学习之最大子序列算法(java实现)
- 一步步学习数据结构和算法之折半插入排序效率分析及java实现
- 【老鸟学算法】大整数乘法——算法思想及java实现
- (数据结构与算法分析 一)------快速求幂算法,Java递归实现
- 一步步学习数据结构和算法之冒泡排序效率分析及java实现
- 链栈实现算法 - Java 学习笔记(26)
- 一步步学习数据结构和算法之快速排序效率分析及java实现
- 【老鸟学算法】大整数乘法——算法思想及java实现
- 黑马程序员java学习—File类,Properties类和递归思想
- 常见算法学习及其Java实现--有序区和无序区的冒泡算法实现
- 每天学习一算法系列(35)(递归和非递归俩种方法实现二叉树的前序遍历)
- 算法学习——LSC最大公有子序列 java实现