原题:洛谷P1562

  这个题的原理和8皇后的原理是一模一样的,就是必须要用n个皇后把每一个行填满,同时满足每一列,每一行,每一条对角线只有一个棋子。但如果按照原来的方法暴打的话只有60分(优化亲测无效)

  所以这个时候,我们可以用二进制来表示一波状态(可以类比状态压缩的二进制)。从上面的条件来看,我们需要表示的量有:行、列、两条对角线(向左的和向右的),我们用一个状态的某一位的1表示这个状态的这个位置不能放(已经有棋子)

  • 对于每一行:

  我们可以用DFS的深度来减少需要表示状态。(也就是说不用管,见代码)

  • 对于每一列:

  因为在DFS的时候,每一行都必然会放一个棋子,此时需要把这一位的列状态置为1,而且由此可知,当最后放满的时候,表示列的那个状态的必然全是1(终止条件

  •  对于向左上的对角线:

  从左上到右下,所以当前这一行影响的应该是下一行的右下一个(↘),然后这里需要注意的是这一行的最后一个的这种对角线是对下一行是没有影响的。举个例子(单就对角线来说):

 这一行:     (1表示有棋子)
下一行:

  这个时候可以看出来,这个状态相当于是>>=1。(因为最后一个没有影响所以它被消掉也没有影响)

  •  对于右上的对角线

  从右上到左下,影响下一行的左一个(↙)其他都和向左上的一样,只是这一行的第一个对下一行没有影响。举个例子:

    这一行:     (1表示有棋子)
下一行:

  这个时候可以看出来,这个状态相当于是<<=1。(第一个会被移到前面,超出查找范围,会被接下来的运算消掉)

  • 由此可见

  对于这一行来说,可以取的点应该是上一行的所有状态求并集(位或)之后二进制状态位为0的数。举个例子:

    上一行传下来的状态:      列:
右对角线:
左对角线:
状态的并集:(这一行可以填充):
那么所以可以放棋子的位置应该是这一行的第2列和第5列

这个时候,为了方便快速找到并集当中的可行解,联想树状数组的lowbit(),发现只要用1来表示可以放的位置即可,具体方法:并集取反后与全部位置(1~n位)都为1的一个值(all)作与运算(与all作与运算是为了把右对角线多向右移出去的除掉)举个例子:

    ~并集:  (右对角线多出两位)
all:
&all:

这样只需要每一次取lowbit()之后把状态更新,并将并集减去lowbit()(将这一位置0表示已经取过)后继续DFS。

代码:(如果还没有理解可以用二进制检验函数单步走一下)

 #include<cstdio>
#include<iostream>
#include<cstring>
using namespace std;
int n,map[]={},all=,cc=;//map[i]存放本来就不能放的点,cc表示可行借个数 //二进制检验函数
int c[]={};
void print_in_2(int x){
for(int i=;i<;i++)c[i]=;
while(x){
c[]++;
c[c[]]=x&;
x>>=;
}
for(int i=n;i>;i--){
printf("%d",c[i]);
}
cout<<endl;
} void Init(){
scanf("%d",&n);
char k[];//给出的地图
for(int i=;i<n;i++){
scanf("%s",k);
for(int j=;j<n;j++){
if(k[j]=='.')
map[i]|=(<<j);//这里是将所给地图左右对称了,易证对称后与原图方案数相同
}
}
all=(<<n)-; /*for(int i=0;i<n;i++){
print_in_2(map[i]);
}*/
}
int low_bit(int x){//返回第一个1
return x&-x;
}
void DFS(int deep,int line,int l_diag,int r_diag){//深度(行数)和上一行的状态:列、左对角线、右对角线
if(line==all){//如果每一列都被填充了,找到答案
cc++;
return;
}
//注意:l_d.r_d,line是以1表示已经取过的位置,不是表示还可以放棋子,所以要与all取反
int may=all&~(map[deep]|line|l_diag|r_diag);//取可行解,由于map中的1只能影响当前行,所以要在取一次或
//print_in_2(may);
int v;
while(may){
v=low_bit(may);
may-=v;//将取出的那一位置0
//print_in_2(may);
DFS(deep+,line+v,(l_diag+v)>>,(r_diag+v)<<);//因为取出的那一位在任意状态中必为0,所以可以直接加上v来把那一位置1;
}
}
void solve(){
DFS(,,,);//我是从0开始存图所以以0为深度开始,其他位置在一开始都可取所以都为0
printf("%d",cc);
}
int main(){
Init();
solve();
return ;
}

题解 洛谷P1562 【还是N皇后】的更多相关文章

  1. 题解 洛谷P5018【对称二叉树】(noip2018T4)

    \(noip2018\) \(T4\)题解 其实呢,我是觉得这题比\(T3\)水到不知道哪里去了 毕竟我比较菜,不大会\(dp\) 好了开始讲正事 这题其实考察的其实就是选手对D(大)F(法)S(师) ...

  2. 题解 洛谷 P3396 【哈希冲突】(根号分治)

    根号分治 前言 本题是一道讲解根号分治思想的论文题(然鹅我并没有找到论文),正 如论文中所说,根号算法--不仅是分块,根号分治利用的思想和分块像 似却又不同,某一篇洛谷日报中说过,分块算法实质上是一种 ...

  3. 题解-洛谷P5410 【模板】扩展 KMP(Z 函数)

    题面 洛谷P5410 [模板]扩展 KMP(Z 函数) 给定两个字符串 \(a,b\),要求出两个数组:\(b\) 的 \(z\) 函数数组 \(z\).\(b\) 与 \(a\) 的每一个后缀的 L ...

  4. 题解-洛谷P4229 某位歌姬的故事

    题面 洛谷P4229 某位歌姬的故事 \(T\) 组测试数据.有 \(n\) 个音节,每个音节 \(h_i\in[1,A]\),还有 \(m\) 个限制 \((l_i,r_i,g_i)\) 表示 \( ...

  5. 题解-洛谷P4724 【模板】三维凸包

    洛谷P4724 [模板]三维凸包 给出空间中 \(n\) 个点 \(p_i\),求凸包表面积. 数据范围:\(1\le n\le 2000\). 这篇题解因为是世界上最逊的人写的,所以也会有求凸包体积 ...

  6. 题解-洛谷P4859 已经没有什么好害怕的了

    洛谷P4859 已经没有什么好害怕的了 给定 \(n\) 和 \(k\),\(n\) 个糖果能量 \(a_i\) 和 \(n\) 个药片能量 \(b_i\),每个 \(a_i\) 和 \(b_i\) ...

  7. 题解-洛谷P5217 贫穷

    洛谷P5217 贫穷 给定长度为 \(n\) 的初始文本 \(s\),有 \(m\) 个如下操作: \(\texttt{I x c}\),在第 \(x\) 个字母后面插入一个 \(c\). \(\te ...

  8. 洛谷P1219 :八皇后(DFS+回溯)

    题目描述 检查一个如下的6 x 6的跳棋棋盘,有六个棋子被放置在棋盘上,使得每行.每列有且只有一个,每条对角线(包括两条主对角线的所有平行线)上至多有一个棋子. 上面的布局可以用序列2 4 6 1 3 ...

  9. 题解 洛谷 P2010 【回文日期】

    By:Soroak 洛谷博客 知识点:模拟+暴力枚举 思路:题目中有提到闰年然后很多人就认为,闰年是需要判断的其实,含有2月29号的回文串,前四位是一个闰年那么我们就可以直接进行暴力枚举 一些小细节: ...

随机推荐

  1. Java中常量的概念

    常量:在程序执行过程中,其值不发生改变的量.分类:A:字面值常量B:自定义常量字面值常量A:字符串常量(用“”括起来的内容).举例:"hello"B:整数常量 (所有的整数)举例: ...

  2. .NET Core下的开源分布式任务调度系统ScheduleMaster-v2.0低调发布

    从1月份首次公开介绍这个项目到现在也快4个月了,期间做了一些修修补补整体没什么大的改动.2.0算是发布之后第一个大的版本更新,带来了许多新功能新特性,也修复了一些已知的bug,在此感谢在博客.Issu ...

  3. 使用openmp进行并行编程

    预处理指令pragma 在系统中加入预处理器指令一般是用来允许不是基本c语言规范部分的行为.不支持pragma的编译器会忽略pragma指令提示的那些语句,这样就允许使用pragma的程序在不支持它们 ...

  4. JavaScript之浅谈内存空间

    JavaScript之浅谈内存空间 JavaScipt 内存自动回收机制 在JavaScript中,最独特的一个特点就是拥有自动的垃圾回收机制(周期性执行),这也就意味者,前端开发人员能够专注于业余, ...

  5. 笔记-VUE滚动加载更多数据

    来源:https://blog.csdn.net/qq_17281881/article/details/87342403 VUE滚动加载更多数据 data() { return { loading: ...

  6. php获取远程文件内容的函数

    一个简单的php获取远程文件内容的函数代码,兼容性强.直接调用就可以轻松获取远程文件的内容,使用这个函数也可获取图片.代码如下: ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 ...

  7. 2019-2020-1 20199310《Linux内核原理与分析》第八周作业

    1.问题描述 在前面的文章中,学习了在Linux系统之中如何创建一个新进程进行追踪,本文将围绕编译链接的过程和ELF可执行文件格式,对Linux内核装载和启动一个可执行程序. 2.解决过程 2.1 E ...

  8. 关于搭建IIS网页弹出登录框的解决方案

    今天自己搭建IIS服务器的时候,明明设置了匿名访问,但是用ie访问127.0.0.1的时候还是会弹出一个登陆框,最后在网上找到答案. 转自: https://blog.csdn.net/sunleib ...

  9. 痞子衡嵌入式:揭秘i.MXRT1170 eFuse空间访问可靠性的保护策略(冗余与ECC)

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是恩智浦i.MXRT1170的eFuse空间访问可靠性保护策略. 关于i.MXRT系列的eFuse/OTP,痞子衡之前在介绍Boot时写过 ...

  10. 【20180129】java进程经常OOM,扩容swap。

    导读:线上一台服务器专门做为公司内部apk打包服务,由于app的业务和功能与时俱增,apk打包需要依赖的资源越来越多,最近这几天每次apk打包的时候都会由于OOM导致打包失败.由于apk打包业务并不是 ...