题解N皇后问题

Posted 2021-yanghaoran

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了题解N皇后问题相关的知识,希望对你有一定的参考价值。

题目描述

相信大家都听过经典的“八皇后”问题吧?这个游戏要求在一个8×8的棋盘上放置8个皇后,使8个皇后互相不攻击(攻击的含义是有两个皇后在同一行或同一列或同一对角线上)。
桐桐对这个游戏很感兴趣,也很快解决了这个问题。可是,他想为自己增加一点难度,于是他想求出n皇后的解的情况。
你能帮助她吗?

输入输出格式

输入格式:

一行,仅有一个数n(1≤n≤14),表示为n皇后问题。

输出格式:

输出仅有一个数,表示n皇后时问题的解法总数。

输入输出样例

输入样例:

8

输出样例:

92
这道题用搜索可以压线过,所以就讲一讲搜索的做法
首先我们想到用二维数组做,但是二维数组肯定会超时,所以我们用降维做法,原先在棋盘上(1,1)位置有棋子,就把数组chessboard[1][1]标记为1
然后我们可以吧它改为(1,1)有棋子就把chessboard[1]标记为1,这样可以减少时间复杂度,13皇后可以压线过
然后我们再优化,用数组计数,我们发现,第i行的棋子会在第\\(x+i\\)条正对角线上,会在第\\(x-i+n\\)条反对角线上,就用数组计数,每访问一次这个对角线,就把它标记为1,即可减少超时
可是这个做法对于14皇后还需要大约两秒,我们再优化
发现这个棋盘具有对称性,以4皇后为例,举例如下
0 1 0 0
0 0 0 1
1 0 0 0
0 0 1 0
如果这个做法可行,那么下列做法
0 0 1 0
1 0 0 0
0 0 0 1
0 1 0 0
也是可行的,所以第一行只需要放到\\((n+1)/2\\)就可以了
至此,基本可以吧14皇后优化至1秒内
具体程序如下:

#include<cstdio>
using namespace std;
int queen[15],n,ans;
int tr1[35],tr2[35],tr3[35];
void dfs(int x)
{
    if(x>n)
    {
        ans+=2;
        return;
    }
    for(register int i=1;i<=n;++i)
    {
        if(tr1[i]==0&&tr2[x+i]==0&&tr3[x-i+n]==0)
        {
            queen[x]=i;
            tr1[i]=1;
            tr2[x+i]=1;
            tr3[x-i+n]=1;
            dfs(x+1);
            tr1[i]=0;
            tr2[x+i]=0;
            tr3[x-i+n]=0;
            queen[x]=0;
        }
    }
}
void newdfs(int x)
{
    if(x>n)
    {
        ans+=2;
        return;
    }
    for(register int i=1;i<=n/2;++i)
    {
        if(tr1[i]==0&&tr2[x+i]==0&&tr3[x-i+n]==0)
        {
            queen[x]=i;
            tr1[i]=1;
            tr2[x+i]=1;
            tr3[x-i+n]=1;
            dfs(x+1);
            tr1[i]=0;
            tr2[x+i]=0;
            tr3[x-i+n]=0;
            queen[x]=0;
        }
    }
}
int main()
{
    scanf("%d",&n);
    if(n==1)
    {
        cout<<1;
        return 0;
    }
    for(register int i=1;i<=n/2;++i)
    {
        queen[1]=i;
        tr1[i]=1;
        tr2[i+1]=1;
        tr3[1-i+n]=1;
        dfs(2);
        tr1[i]=0;
        tr2[i+1]=0;
        tr3[1-i+n]=0;
    }
    if(n%2==1)
    {
        queen[1]=(n/2+1);
        tr1[(n/2+1)]=1;
        tr2[(n/2+1)+1]=1;
        tr3[1-(n/2+1)+n]=1;
        newdfs(2);
    }
    printf("%d",ans);
}

参考文献:N皇后的多种巧妙解法

以上是关于题解N皇后问题的主要内容,如果未能解决你的问题,请参考以下文章

皇后游戏 题解

1295 N皇后问题

n皇后问题

n皇后问题

codevs 1295 N皇后问题

[HDU2553]N皇后问题(DFS)