N皇后问题(暴力dfs)

题目: 
N皇后在一个N*N的棋盘上,使其不能相互攻击,即任意两个皇后不得处于同一行,同一列或一条对角线上。

(皇后的攻击是米字型,即不能同行不能同列,不能同对角线)

 

思路:

我们先假设n为一个小一点的数字,让问题看起来简单些,这里假设n==4;

 一行不可能放入两个皇后,  而所有的皇后都要放进去, 所以每个皇后都在不同的行,每行必有一个皇后

我们从第一行开始,每取出一个皇后,放入一行,共有4种不同的放法;

然后再放第二个皇后,还是有4种不同的放法

此时就是一个四叉树的结构

 于是我们可以用一个四叉树来描述这个过程。从根节点开始,树每增加一层,便是多放一个皇后,

 直到第四层(根节点为0层),最后得到一个完全四叉树。  

紧接着我们开始用深度优先遍历这个四叉树,在遍历的过程中,进行相应的条件的判断。以便去掉不合规则的子树。

    那么具体用什么条件来进行剪枝呢?

    我们先对问题解的结构做一个约定。

    用a[i]来表示 ,a[i]为第i行皇后所在的列数 (a实际为一种映射关系)

这时我们考虑第一个条件,不能在同一行,同一列于是我们得到a[i]不能相同。

剩下一个条件是不能位于对角线上,这个条件不是很明显,但我们知道对角线的45°即斜率为1;

那么设两个不同的皇后分别在i,j行上,a[i],a[j]分别表示在i,j行哪一列上。

那么在同一对角线的条件可以写为abs((j-i))==abs(a[i]-a[j]),其中abs为求绝对值的函数(c++在<cmath>的头文件里>

#include <iostream>
#include <cmath>
using namespace std;
#define N 8
int sum=0;
int a[N+1]; //a为层数到列数的映射关系
void  dfs(int ceng)
{
    if(ceng==N+1)
    {
        for(int i=1;i<ceng;i++)  //这里的比较很巧妙,比如层数是4,这里就是1,2和1比,3和2和1比,4和3和2和1比;这样就保证了任意两层都比较过了
            for(int j=1;j<i;j++)
                if(a[j]==a[i]||abs(i-j)==abs(a[i]-a[j]))
                   return;
        sum++;
        return; //注意这里两种情况都要return; 不然会卡死在这个递归里
    }
    for(int i=1;i<=N;i++)
    {
        a[ceng]=i;
        dfs(ceng+1);
    }
    return;
}
int main()
{
    dfs(1); //从第一层开始放
    cout<<sum;
    return 0;
}

0-0然后就可以了

这个方法是纯暴力,把所有的可能情况都试了一遍,所以极其慢,要过hdu的话需要打表骗分

后来想了下,这道题用全排列可能写起来更容易些,

嘛,电脑快没电了,宿舍还没来电,

等来电了再补上吧,

溜了溜了

posted @ 2018-03-23 11:59  uzi2008  阅读(1181)  评论(0编辑  收藏  举报