DFS & 位运算 | 洛谷P1219 八皇后

题目描述

检查一个如下的6 x 6的跳棋棋盘,有六个棋子被放置在棋盘上,使得每行、每列有且只有一个,每条对角线(包括两条主对角线的所有平行线)上至多有一个棋子。

DFS & 位运算 | 洛谷P1219 八皇后_第1张图片

上面的布局可以用序列2 4 6 1 3 5来描述,第i个数字表示在第i行的相应位置有一个棋子,如下:

行号 1 2 3 4 5 6

列号 2 4 6 1 3 5

这只是跳棋放置的一个解。请编一个程序找出所有跳棋放置的解。并把它们以上面的序列方法输出。解按字典顺序排列。请输出前3个解。最后一行是解的总个数。

DFS & 位运算 | 洛谷P1219 八皇后_第2张图片

题目链接:https://www.luogu.org/problemnew/show/P1219

方法一:

DFS + 对角线的优化
利用一次函数y=kx+b,因为b为常数,利用b来代表一整条对角线。

DFS & 位运算 | 洛谷P1219 八皇后_第3张图片
对角线优化的说明

AC代码如下:
(代码有具体的解释)

//【DFS】 
//【对角线的处理】 
//用时: 560ms / 内存: 820KB

#include
#include
using namespace std;

int n;
int r[400]; //存储第i的皇后的列的位置
int c[400]; // column[j]
int z1[400]; //左上到右下的对角线 (ld)
int z2[400]; //左下到右上的对角线 (rd)
int total=0;

void print()
{
    if(total<=2) //total=0,1,2[三个解]
    {
        for(int i=1;i<=n;i++)
        {
            printf("%d ",r[i]); //输出结果
        }
        printf("\n");
    }
    total++; //解+1
}

void dfs(int row) //DFS[核心代码]
{
    if(row>n) //找到解
    {
        print();
        return;
    }
    
    for(int j=1;j<=n;j++)
    {
        if(c[j]==0 && z1[j+row]==0 && z2[j-row+n]==0) //同列、同对角线无皇后
        {
            r[row]=j; //在该列放置第row
            c[j]=1; //该列置1
            z1[j+row]=1; // ld对角线置1
            z2[j-row+n]=1; //rd对角线置1
            dfs(row+1); //深度向下搜索【递归】

            // 恢复原来的状态【回溯】
            c[j]=0; 
            z1[j+row]=0;
            z2[j-row+n]=0;
        }
    }
} 

int main()
{
    scanf("%d",&n);
    dfs(1); //从第一位皇后开始
    printf("%d",total);
    return 0;
} 

方法二:

利用位运算进行优化
(弱弱的我对于位运算也不是特别熟悉)

//【位运算优化】 
//用时: 112ms / 内存: 804KB
#include
#include
#include
using namespace std;

int n;
int upperlim,row,ld,rd;
int res[25]; //存储一个解的结果 
//int all[4][25]; //存储结果【所有】 
int ans=0; //可能的解 
int sum=0;

void dfs(int row,int ld,int rd)
{
    int pos,p;
    if(row!=upperlim) //未找到解 
    {
        pos=upperlim&(~(row|ld|rd)); //获取可以放置皇后的所有位置(row|ld|rd获取所有的禁止位置,再取反即可得到可以放置的位置)
        while(pos) //pos!=00000000【n个0】
        {
            p=pos&(-pos); //取出最右边的1【一个可能的位置】
            pos=pos-p; //将pos最右边为1的bit清零,也就是为获取下一次的最右可用列使用做准备,
                       //程序将来会回溯到这个位置继续试探 
            
            int num=p;
            res[sum++]=__builtin_ffs(num);  // 【从右往左数获取皇后所在的列数。因为棋盘对称,所以从左往右数于从右往左数结果一致】
            //int __builtin_ffs(unsigned int x)
            //返回二进制表示中x的最后一位1(最右边的)是从后向前第几位,比如 7368(1110011001000)返回 4
            dfs(row|p,(ld|p)<<1,(rd|p)>>1);  //
            res[--sum]=0; //回溯【***重要***】 
        } 
    }
    else //找到一个解 
    {
        ans++; //解+1
        
        if(ans<=3)
        {
           for(int i=0;i

C++高效位运算:

int __builtin_ffs(unsigned int x)

返回二进制表示中x的最后一位1(最右边的)是从后向前第几位,比如 7368(1110011001000)返回 4

int num=p;
res[sum++]=__builtin_ffs(num); 

等价于

int num=p;
int count=0;
while(num)
{
     num=num>>1; //右移一位【等同于除2】 
     count++; //加1
}
res[sum++]=count;
参考资料:

位运算优化:
https://blog.csdn.net/kai_wei_zhang/article/details/8033194
https://blog.csdn.net/Dora_Bin/article/details/52733832?utm_source=blogxgwz5

C++高效位运算:
https://www.cnblogs.com/P6174/p/8759249.html

写在最后:

打一波广告,自己的公众号,不是技术文,主要是分享自己的一些想法,欢迎前来关注,非喜勿喷。


DFS & 位运算 | 洛谷P1219 八皇后_第4张图片
我锨说

你可能感兴趣的:(DFS & 位运算 | 洛谷P1219 八皇后)