Java与算法之(6) - 八皇后问题

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

(文字和图片来自百度百科)

如果动手来摆放皇后,可以用这样一种思路:在最左侧一列放下一个皇后,然后在右边一列从上到下找到第一个与左边皇后不冲突的位置,摆放第二个皇后;再向yo一列,从上到下找到第一个与前两个皇后不冲突的位置摆放第三个皇后,依次类推,直到在最后一列摆下第八个皇后。

认真思考的话,可以发现这仍然是深度优先搜索的思路,即步步推进,下一步做的事情和当前是一样的。代码:

[java] view plain copy

print?

  1. public class DfsEightQueens {
  2. int[] queens = new int[8]; //记录每一列皇后的摆放位置
  3. int count = 0; //摆法总数
  4. public void dfs(int column) {
  5. if(column == 8) { //8个皇后都已经摆放
  6. count++;
  7. System.out.println("第" + count + "种方法:");
  8. print();
  9. return;
  10. }
  11. for(int i = 0; i < 8; i++) {
  12. queens[column] = i; //在该列的第i行上放置皇后
  13. if(isValid(column)) //检查摆放在该位置是否与前column-1列的皇后有冲突
  14. dfs(column + 1); //没有冲突则开始下一列8个位置的尝试
  15. }
  16. }
  17. private boolean isValid(int column) {
  18. for(int i = 0; i < column; i++) { //第column列上的皇后与前面column-1个皇后比较
  19. if(queens[i] == queens[column]) //两个皇后在同一行上
  20. return false;
  21. if(Math.abs(queens[i] - queens[column]) == (column - i)) //两个皇后在同一对角线上
  22. return false;
  23. }
  24. return true;
  25. }
  26. private void print() {
  27. for(int i = 0; i < 8; i++) {
  28. for(int j = 0; j < 8; j++) {
  29. if(queens[i] == j)
  30. System.out.print("* ");
  31. else
  32. System.out.print("_ ");
  33. }
  34. System.out.println();
  35. }
  36. }
  37. public static void main(String[] args) {
  38. DfsEightQueens q = new DfsEightQueens();
  39. q.dfs(0);
  40. System.out.println("共" + q.count + "种摆放方法");
  41. }
  42. }

输出:

[java] view plain copy

print?

  1. 共92种摆放方法
时间: 2024-09-30 06:28:36

Java与算法之(6) - 八皇后问题的相关文章

算法题(八皇后)

问题: 八皇后问题,是一个古老而著名的问题,是回溯算法的典型案例.该问题是国际西洋棋棋手马克斯·贝瑟尔于1848年提出:在8×8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行.同一列或同一斜线上,问有多少种摆法? 代码: public class huanghou{ private int[] column;//同栏是否有皇后,1表示有 private int[] rup;//右上至左下是否有皇后 private int[] lup;//左上至右下是否有皇后 pri

回溯算法--八皇后问题

问题描述:八皇后不能相互攻击,即,八个皇后不能同行,同列,不同在同一条对角线上, 对角线又可以分为左对角线和右对角线 左对角线上满足:i-j+7都相等(i,j分别是一维和二维的坐标) 右对角线满足:i+j 都相等 代码如下: #include <iostream> using namespace std; int e_q[8][8]; int r[15]; int le[15]; int q[8]; int count = 0; void en_queens(int); int main()

JAVA常见算法题(十八)

package com.xiaowu.demo; /** * 两个乒乓球队进行比赛,各出三人.甲队为a,b,c三人,乙队为x,y,z三人,以抽签决定比赛名单. 有人向队员打听比赛的名单:a说他不和x比,c说他不和x. * z比.请编程序找出三队赛手的名单. * * @author WQ * */ public class Demo18 { public static void main(String[] args) { vs(); } public static void vs() { char

回溯算法解八皇后问题(java版)

八皇后问题是学习回溯算法时不得不提的一个问题,用回溯算法解决该问题逻辑比较简单. 下面用java版的回溯算法来解决八皇后问题. 八皇后问题,是一个古老而著名的问题,是回溯算法的典型案例.该问题是国际西洋棋棋手马克斯·贝瑟尔于1848年提出:在8×8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行.同一列或同一斜线上,问有多少种摆法. 思路是按行来规定皇后,第一行放第一个皇后,第二行放第二个,然后通过遍历所有列,来判断下一个皇后能否放在该列.直到所有皇后都放完,或者放哪

回溯法解决八皇后问题

八皇后问题是学习回溯算法时不得不提的一个问题,用回溯算法解决该问题逻辑比较简单. 下面用java版的回溯算法来解决八皇后问题. 八皇后问题,是一个古老而著名的问题,是回溯算法的典型案例.该问题是国际西洋棋棋手马克斯·贝瑟尔于1848年提出:在8×8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行.同一列或同一斜线上,问有多少种摆法. 思路是按行来规定皇后,第一行放第一个皇后,第二行放第二个,然后通过遍历所有列,来判断下一个皇后能否放在该列.直到所有皇后都放完,或者放哪

c++ 八皇后问题(转载)

转自 雪狼的程序故事 http://www.cnblogs.com/gaoteng/archive/2012/04/11/2442692.html 经典回溯算法(八皇后问题) 今天偶尔看到了一个算法问题(八皇后问题),回想一下还是在算法课上学习过的,于是,自己总结了一下,写了这篇日志 算法提出: 在国际象棋棋盘上(8*8)放置八个皇后,使得任意两个皇后之间不能在同一行,同一列,也不能位于同于对角线上.问共有多少种不同的方法,并且指出各种不同的放法. 算法思路: 首先我们分析一下问题的解,我们每取

八皇后(JAVA算法实现)

在学习现代软件工程构建之法这门课时,老师要求发表一篇博客,使用JAVA算法实现八皇后问题的求解.写这篇博客时,我学习了一些其他的博客,因为我常常遇到问题,自己无法解决时,向他人学习也是一种方法. 国际西洋棋棋手马克斯·贝瑟尔于1848年提出:在8×8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行.同一列或同一斜线上,问有多少种摆法. 我们可以逐行或者逐列来进行可行摆放方案的遍历,每一行(或列)遍历出一个符合条件的 位置,接着就到下一行或列遍历下一个棋子的合适位置,这

八皇后问题求解java(回溯算法)

八皇后问题 八皇后问题,是一个古老而著名的问题,是回溯算法的典型案例.该问题是国际西洋棋棋手马克斯·贝瑟尔于1848年提出:在8×8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行.同一列或同一斜线上,问有多少种摆法. 代码实现 static int count = 0;//记录有几种方法 int max = 8;//max表示几个皇后 int[] arr = new int[max];//用一个数组表示,其中arr[n]表示放在第arr[n]+1列,n表示第n+1个

简述java递归与非递归算法,0-100求和,斐波那契数列,八皇后,汉诺塔问题

一:什么是递归算法? 递归算法就是直接或者间接的调用自己的方法,在达到一个条件的时候停止调用(递归出口),所以一定要找准好条件,让递归停止,否则就会是无限进行下去 二:递归程序设计的关键 1:找出调用中所需要的参数 2:返回的结果 3:递归调用结束的条件 三:递归程序注意 1:要有方法中自己调用自己 2:要有分支结构 3:要有结束的条件 四:简单叙述递归函数的优缺点 优点: 1:简洁清晰,实现容易,可读性好 2:在遍历的算法中,递归比循环更为简单 缺点: 1:效率低,使用递归函数是有空间和时间的