当前位置:首页 > 题解目录 > 正文内容

八皇后问题

亿万年的星光5年前 (2021-01-28)题解目录4622

【题目描述】

八皇后问题是一个古老而著名的问题,是回溯算法的典型例题。该问题是十九世纪著名的数学家高斯1850年提出:在8X8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行、同一列或同一斜线(对角线)上,问有多少种摆法。

【输入描述】

【输出描述】

一共有多少种摆法

【样例输入】

【样例输出】

92

【题目分析】

(1)框架分析

放置第i个(行)皇后的算法为:
int search(i);
 {
     int j;
   for (第i个皇后的位置j=1;j<=8;j++ )	    //在本行的8列中去试
   if (本行本列允许放置皇后)
    {
     放置第i个皇后;
                  对放置皇后的位置进行标记;
     if (i==8) 输出	                              //已经放完个皇后
        else search(i+1);                //放置第i+1个皇后
     对放置皇后的位置释放标记,尝试下一个位置是否可行;
    }
 }

(2)

主要解决几个问题:
冲突:包括行、列、两条对角线
行:规定每行放一个皇后,不会造成行上的冲突
列:当第col列被某个皇后占领后,则同一列上所有位置不在放皇后,要把    flag[col] 置为被占领状态。
对角线:对角线有两个方向。当第n行第col列皇后占领后,上下对角线标记               为被占领状态。

规律:左上到右下,我们叫主对角线,同一主对角线i-j是个常数。右上到左下,我们称为副对角线,同一副对角线i+j是个常数。所以,

左上到右下:   i-j+7  (C++ 不支持负数)
右上到左上:   i+j

(3)

1.初始化(清除棋盘)
2.循环8次
      放置一个皇后;      
      检查是否满足条件,如满足,登记皇后位置;      
      如不满足,则退回,增加一步后在放皇后。
3. 知道放置最后一个皇后

【参考代码1】

用col表示列,L_diagonal表示主对角线(就是左上到右下),R_diagonal表示副对角线(就是右上到左下)

#include<iostream>
using namespace std;
int col[10],L_diagonal[10],R_diagonal[10]; //分别表示标记数组,列数组,主对角线数组,副对角线数组
int ans[10]; //结果数组
int sum;
int dfs(int i)//用i来表示皇后所处的行位置
{
   int j; //用j表示皇后所处的列位置
   for(j=1;j<=8;j++)
       if((col[j]==0)&&(L_diagonal[i-j+7]==0)&& (R_diagonal[i+j]==0))// 列、主对角线、副对角线没有被占领过
       {
           ans[i]=j; //在当前(i,j)位置上摆放一个皇后
           col[j]=1; //占领当前(i,j)所在的列
           L_diagonal[i-j+7]=1; //占领当前(i,j)所在的主对角线
           R_diagonal[i+j]=1; //占领当前(i,j)所在的副对角线
           if(i==8)
           {
              //print(); //此题没有要求输出
               sum++; //摆放种数加1    
           }
           else
               dfs(i+1);
           col[j]=0; //回溯(i,j)所在的列
           L_diagonal[i-j+7]=0; //回溯当前(i,j)所在的主对角线
           R_diagonal[i+j]=0;//回溯当前(i,j)所在的副对角线    
       }
}
int main()
{
   dfs(1);
   cout<<sum<<endl;
   return 0;
}


扫描二维码推送至手机访问。

版权声明:本文由青少年编程知识记录发布,如需转载请注明出处。

分享给朋友:
返回列表

上一篇:数的拆分(1)

下一篇:八皇后2

相关文章

数的拆分(1)

【题目描述】任何一个大于1的自然数n,总可以拆分成若干个小于n的自然数之和。例如:当n=7时7=1+1+1+1+1+1+1 7=1+1+1+1+1+2 7=1+1+1+1+3 7=1+1+1+2...

【题解】尼科彻斯定理

【题目描述】 验证尼科彻斯定理,即:任何一个整数m的立方都可以写成m个连续奇数之和。【输入描述】任一正整数【输出描述】该数的立方分解为一串连续奇数的和【样例输入】13【样例输出】13*13*...

【题解】最大子矩阵

【题目描述】已知矩阵的大小定义为矩阵中所有元素的和。给定一个矩阵,你的任务是找到最大的非空(大小至少是1 × 1)子矩阵。比如,如下4 × 4的矩阵0  -2 -7&nb...

【题解】最小新整数

【问题描述】第⼀⾏有x个正整数a1,a2,..,ax,第⼆⾏有y个正整数b1,b2,...,by,第三⾏有z个正整数c1,c2,...,cz,假设第⼀⾏的x个正整数中的最⼤值为a、第⼆⾏的y个正整数中...

【题解】小X与机器人

【题解】小X与机器人

【题目描述】小X的老师很喜欢围棋。众所周知,围棋的棋盘有19行19列,共有361个交叉点。为方便起见,我们把这些行列按顺序编号为1~19,并用(x, y)表示第x列第y行的位置。例如下图中,A用(16...

【题解】大整数乘法

【题目描述】求两个不超过200位的非负整数的积。【输入描述】有两行,每行是一个不超过200位的非负整数,没有多余的前导0。【输出描述】一行,即相乘后的结果。结果里不能有多余的前导0,即如果结果是342...