回溯法与N皇后问题JAVA实现

标签: 回溯算法 算法 N皇后 Java
54人阅读 评论(0) 收藏 举报
分类:

提出问题

八皇后问题:在8*8格的国际象棋上摆放8个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行、同一列或同一斜线上,问有多少种摆法。

如何解决这个问题?一种常用且有效的方法是回溯法,是用树形结构描述问题的求解过程。

什么是回溯法

回溯法是一类非常重要的算法设计方法,有“通用解题法”之称。

回溯法(探索与回溯法):一种选优搜索法,又称试探法。利用试探性的方法,在包含问题所有解的解空间树中,将可能的结果搜索一遍,从而获得满足条件的解。搜索过程采用深度遍历策略,并随时判定结点是否满足条件要求,满足要求就继续向下搜索,若不满足要求则回溯到上一层,这种解决问题的方法称为回溯法。

回溯法解求解问题步骤
  1. 针对给定问题,定义问题的解空间树
  2. 确定易于搜索的解空间结构
  3. 以深度优先方式搜索解空间,并且在搜索过程中永剪枝函数避免无效搜索。

用回溯法求解问题,重点是设计问题的解空间树,其解题过程则是深度遍历解空间树的过程

解空间树:是依据待求解问题的特性,用树结构表示问题的解结构、用叶子表示问题所有可能的解的一棵树。

解空间树的形成过程:我们可以把求解问题的过程当作一系列的决定来考虑,回溯法对每一个决定都系统地分析所有可能的结果。而每一次决定即为解空间树中的一个分支结点,各种可能的结果便形成了各棵不同的子树,问题最终所有可能的解将展现在所有的叶子上。这便是解空间树的形成过程。

对解空间树的遍历可搜索到问题所有可能的解,因此,可获得满足要求的全部解,也可通过对所有解的比较选择获得最优解。

由于空间问题,下面给出一个三皇后问题的解空间树(3皇后问题无解),树中第i层的结点决定第i行皇后的摆放位置,均有三种不同的选择,便形成了三个孩子结点,但其中不包括不符合要求的布局。N皇后问题解空间树与三皇后问题解空间树类似。


求解N皇后问题的回溯法

N皇后问题要求求解在N*N的棋盘上放置N个皇后,并使各皇后彼此不受攻击的所有可能的棋盘布局。皇后彼此不受攻击的约束条件是:任何两个皇后均不能在棋盘上同一行、同一列或者同一对角线上出现。

由于N皇后问题不允许两个皇后在同一行,所以,可用一维数组X表示N皇后问题的解,X[i]表示第i行的皇后所在的列号。例如一个满足要求的四皇后棋盘布局如下图所示,其结果X数组的值为:[2, 4, 1, 3]。


由上述X数组求解N皇后问题,保障了任意两个皇后不在同一行上,而判定皇后彼此不受攻击的其他条件,可以描述如下:

(1)X[i] = X[s],则第i行与第s行皇后在同一列上。

(2)如果第i行的皇后在第j列,第s行皇后在第t列,即X[i] = j和X[s] = t,则只要i-j = s-t或者i+j = s+t,说明两个皇后在同一对角线上。

对两个等式进行变换后,得到结论:只要|i-s| = |j-t|(即i-s = X[i]-X[s]),则皇后在同一对角线上。

解N皇后问题需要遍历解空间树,遍历中要随时判定当前结点棋盘布局是否符合要求,符合要求则继续向下遍历,直至判断得到一个满足约束条件的叶子结点,从而获得一个满足要求的棋盘布局;不符合要求的结点将被舍弃(称之为剪枝),并回溯到上一层的结点继续遍历。当整棵树遍历结束时,已获得所有满足要求的棋盘布局。

综上所述,用回溯法递归遍历解空间树,求解N皇后问题的算法如下(Java)。

/*
 * 求解N皇后问题的回溯算法
 * 棋盘为n*n,函数从第index行起求解皇后的布局,本函数初始调用为tria(1, n)
 */
public void tria(int index, int n) {
	if(index > n) {
		// TODO 输出X数组(也可以改写算法,比如记录满足约束的棋盘个数)
	} else {
		//依次生成各孩子结点
		for(int j = 1; j <= n; j++) {
			//第index行的皇后放入第j列
			X[index] = j;
			if(place(index)) {
				tria(i+1, n);
				//结点满足约束条件,则递归进入下一层继续遍历,否则跳过
			}
		}
	}
}

/*
 * 皇后位置满足约束条件的判定函数
 */
private boolean place(int s) {
	//判定s行X[s]位置上的皇后,与1至s-1行上各皇后的位置是否满足约束条件
	for(int i = 1; i < s; i++) {
		if((X[i] == X[s]) || (Math.abs(i-s) == Math.abs(X[i]-X[s]))) {
			return false;
		}
	}
	
	return true;
}

求解八皇后问题代码实现

public class Main2 {
	static int resultCount = 0;
	
	private static boolean place(int[] arr, int s) {
		for(int i = 0; i < s; i++) {
			if((arr[i] == arr[s]) || (Math.abs(i-s) == Math.abs(arr[i]-arr[s]))) {
				return false;
			}
		}
		
		return true;
	}
	
	public static void tria(int[] arr, int i, int n) {
		if(i >= n) {
			++resultCount;
		} else {
			for(int j = 0; j < n; j++) {
				arr[i] = j;
				if(place(arr, i)) {
					tria(arr, i+1, n);
				}
			}
		}
	}

	public static void main(String[] args) {
		int[] queen = new int[8];
		tria(queen, 0, 8);
		
		System.out.println(resultCount);
	}
}

运行结果


参考资料:《数据结构与算法》    王曙燕    主编    王春梅    副主编    人民邮电出版社


查看评论

回溯法解决N皇后问题(java实现)

回溯法解决N皇后问题(java实现)
  • qq_22222499
  • qq_22222499
  • 2017-04-30 13:26:13
  • 884

回溯法解决N皇后问题 Java代码实现

  • 2013年04月29日 22:08
  • 183KB
  • 下载

JAVA实现N皇后问题(回溯法)

JAVA实现N皇后问题(回溯法)
  • tyzttzzz
  • tyzttzzz
  • 2015-07-23 20:43:17
  • 3607

算法分析与设计回溯法之n皇后问题

问题介绍:在n*n格的棋盘上放置彼此不受攻击的n个皇后。由于皇后可以攻击与之处于同一行或同一列或在同一斜线上的棋子。n皇后问题等价于在n*n的棋盘上放置n个皇后,任何两个皇后不放在同一列或同一行或同一...
  • hello_tomorrow_111
  • hello_tomorrow_111
  • 2017-12-01 22:17:16
  • 177

n皇后问题—回溯法 C++实现

#include #include using namespace std; bool isLegal(int row,int col,vector &v,int n) { for(int i=...
  • gl486546
  • gl486546
  • 2017-09-02 22:18:13
  • 300

n-皇后问题 C++实现 回溯法

该程序输出所有可行解: /* Function : n-皇后问题的回溯算法 * Author : wyh7280 * Time : 2015.05.20 10:36:0...
  • wyh7280
  • wyh7280
  • 2015-05-20 11:03:27
  • 2354

回溯法及N皇后问题

回溯法是一种系统地搜索问题解答的方法。在搜索的过程中尝试找到问题的解,如果发现找不到了,就退一步,往上回溯(剪枝过程)。对于许多复杂问题和大规模问题都可以使用回溯法。 回溯法的基本思想是按照深度优先...
  • charles1e
  • charles1e
  • 2016-06-12 15:55:23
  • 1791

回溯法求解N皇后问题(Java实现)

转载地址:http://haolloyin.blog.51cto.com/1177454/353105/ 回溯法:也称为试探法,它并不考虑问题规模的大小,而是从问题的最明显的最小规模开始逐步求解...
  • u011918382
  • u011918382
  • 2016-08-17 10:49:19
  • 349

【算法分析】回溯法解八皇后问题(n皇后问题)

回溯法解题思路: (1)针对所给问题,定义问题的解空间;    (2)确定易于搜索的解空间结构;    (3)以深度优先方式搜索解空间,并在搜索过程中用剪枝函数避免无效搜索。八皇后问题: 八...
  • sinat_33052719
  • sinat_33052719
  • 2016-05-18 21:26:05
  • 3303

回溯法解决2n皇后(8皇后)问题

8皇后问题是算法入门的经典,在8*8的国际象棋上摆放八个皇后,使其不能相互攻击,即任意两个皇后都不能处于同一 行,同一列,或者同一斜线上。关于8皇后的解法请见:http://www.cnblogs.c...
  • Bazingaea
  • Bazingaea
  • 2017-02-06 17:15:46
  • 985
    个人资料
    等级:
    访问量: 1297
    积分: 233
    排名: 31万+
    文章存档