美文网首页
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

    写在最后:

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


    我锨说

    相关文章

      网友评论

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

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