美文网首页
DFS & 位运算 | 洛谷P1219 八皇后

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

作者: 0与1的邂逅 | 来源:发表于2018-12-19 17:01 被阅读0次

题目描述

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

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

行号 1 2 3 4 5 6

列号 2 4 6 1 3 5

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

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

方法一:

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

对角线优化的说明

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

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

#include<iostream>
#include<cstdio>
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<iostream>
#include<cstdio>
#include<cstring>
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<n;i++) //n个皇后 
           {
              printf("%d ",res[i]);
           }
           printf("\n"); 
        }
    } 
}

int main()
{
    scanf("%d",&n);
    upperlim=(1<<n)-1; //生成n个1组成的二进制数,每一个二进制位(后者说是1)都代表一位皇后
    dfs(0,0,0); //从0,0,0开始搜索
    printf("%d",ans);
    return 0;
}

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

写在最后:

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


我锨说

相关文章

  • 洛谷P1219八皇后-dfs

    题目传送:洛谷P1219八皇后 dfs

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

    题目描述 检查一个如下的6 x 6的跳棋棋盘,有六个棋子被放置在棋盘上,使得每行、每列有且只有一个,每条对角线(包...

  • 【洛谷 P1219】八皇后

    八皇后(题目链接) 思路 典型的深搜回溯问题 代码

  • 位运算及其应用

    内容概要: 位运算基本操作 基于位运算的状态压缩 位运算经典应用 位运算解N皇后问题 位运算 符号描述规则&与1&...

  • N-Queens I & II (Leetcode 51, 52

    著名的“N皇后”问题。主要思路是:我们逐行进行DFS,而在每一行的DFS中逐列扫描,放置皇后。 N Queens ...

  • DFS与N皇后问题

    DFS与N皇后问题 DFS 什么是DFS DFS是指深度优先遍历也叫深度优先搜索。 它是一种用来遍历或搜索树和图数...

  • 单词接龙 DFS 洛谷P1019

    主要是字符串拼接处理比较麻烦(下见mix函数),DFS还是比较简单的 鉴定 读题有坑 注意max_初始化 短小精悍...

  • dfs回潮、标记解决八皇后问题

    题目描述 检查一个如下的6 x 6的跳棋棋盘,有六个棋子被放置在棋盘上,使得每行、每列有且只有一个,每条对角线(包...

  • 洛谷P3374线段树 单点增减 区间求和

    题目传送 洛谷P3374 左移运算符用“<<”表示,是将运算符左边的对象,向左移动运算符右边指定的位数,并且在低位...

  • 【洛谷】DP-过河卒

    一、题目 二、做题总结 本题之前在ZSC上是做过的,当初用的是DFS深度搜索,这次在洛谷上还是原来的思路,却被提示...

网友评论

      本文标题:DFS & 位运算 | 洛谷P1219 八皇后

      本文链接:https://www.haomeiwen.com/subject/ugyjkqtx.html