回溯法求N皇后问题

 回溯法:是一种选优搜索法, 回溯法从开始节点(根节点)出发,以深度优先方式搜索整个解空间。但当探索到某一步时,发现原先选择并不优或达不到目标,就退回一步重新选择,这种走不通就退回再走的技术为回溯法,而满足回溯条件的某个状态的点称为“回溯点”。

N皇后问题描述:在n乘n格的棋盘上放置彼此不受攻击的n个皇后。按照国际象棋的规则,皇后可以攻击与之在同行或同列或同一斜线上的棋子。n后问题等价于在n 乘 n格的棋盘上放置n个皇后,任何2个皇后不放在同一行或同一列或同一斜线上。

求解思路有序地从第 1 行的第 1 列开始,尝试放上一个皇后,然后再尝试第 2 行的第几列能够放上一个皇后,如果第 2 行也放置成功,那么就继续放置第 3 行,如果此时第 3 行没有一行可以放置一个皇后,说明目前为止的尝试是无效的(即不可能得到最终解),那么此时就应该回溯到上一步(即第 2 步),将上一步(第 2 步)所放置的皇后的位置再重新取走放在另一个符合要求的地方…如此尝试性地遍历加上回溯,最后得到和规则的解。

package 算法实验;
/**
 * n皇后类
 *  回溯法求n皇后
 *
 */
public class NQueens {
	private static int queensNum ;//皇后的个数
	//定义一个数组,x[k]表示第k行的列数,x[k]=j表示第k行的第j位置放一位皇后
	private static int []x;
	/**
	 * 产生n位皇后的方法
	 */
	public void produceNQueens(){
		x[1]=0;//每一行从0开始
		int k=1;//从第1行开始
		while(k>0){
			x[k]=x[k]+1;
			while((x[k]<=queensNum)&&!(place(k)))//判断约束条件
				x[k]=x[k]+1;
			if(x[k]<=queensNum)
				if(k==queensNum)
				{
					//输出各行皇后的位置
					for(int i=1;i<=queensNum;i++)
					System.out.print("("+i+","+x[i]+") ");
					System.out.println();
				}
				else {
					k++;
					x[k]=0;
				}
			else k--;//回溯
		}
			
	}
	/**
	 * 判断放置皇后的约束条件的方法
	 * @param k
	 * @return
	 */
	public boolean place(int k){
		for(int j=1;j<k;j++)
			//任何2个皇后不能放在同一行或同一列或同一斜线上。
			if((Math.abs(j-k))==(Math.abs(x[j]-x[k]))||(x[j]==x[k]))return false;
		return true;
		
	}
	/**
	 * 程序入口:主函数
	 * @param args
	 */
	public static void main(String[] args){
		 queensNum=4;//给皇后个数初始值
		x = new int[queensNum+1];//给数组x赋长度
		//实例化对象
		NQueens nq = new NQueens();
		System.out.println(+queensNum+"位皇后的位置是:");
		//调用产生n位皇后的方法
		nq.produceNQueens();
	}

}

 

当N=4时,运行结果是:

 4位皇后的位置是:
(1,2) (2,4) (3,1) (4,3)
(1,3) (2,1) (3,4) (4,2)

 

 当N=6时,运行结果是:

 6位皇后的位置是:
(1,2) (2,4) (3,6) (4,1) (5,3) (6,5)
(1,3) (2,6) (3,2) (4,5) (5,1) (6,4)
(1,4) (2,1) (3,5) (4,2) (5,6) (6,3)
(1,5) (2,3) (3,1) (4,6) (5,4) (6,2)

 

 

 

你可能感兴趣的:(java,回溯法,n皇后问题)