原题:洛谷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. 前端面试的那些事儿(1)~JavaScript 原始数据类型

    前言 自我总结面试常问的一些细节,方便不断回顾与补充.第一次发表文章,如有问题或不足之处望及时指出. JavaScript 原始数据类型 1.1 基础数据类型 7大基础数据类型 boolean nul ...

  2. python中文语料分词处理,按字或者词cut_sentence

    cut_sentence.py import string import jieba import jieba.posseg as psg import logging #关闭jieba日制 jieb ...

  3. 双系统情况下,ubuntu开机挂载Windows分区

    首先:blkid,查看分区所属uuid 其中 /dev/sda5 就是Windows分区 其次:fdisk -l,查看分区情况 通过硬盘大小找到对应要设置的具体分区(其实这步也不用,我只是为了确定) ...

  4. git工具上传项目到码云

    首先,你需要在本地安装git客户端,此处简单易懂,略过然后,在本地建好文件夹,以本人为例,我的路径为 E:\git_project,此时需要通过鼠标右键选择:git bush here 如图所示然后会 ...

  5. linux 磁盘存储和文件系统

    1.磁盘结构 设备文件:关联至一个设备驱动程序,进而能够与之对应硬件设备进行通信 I/O Ports:I/O 设备地址 一切皆文件:open(),read(),write(),close() 设备类型 ...

  6. php下载各种编辑器输出的内容到word中展示

    <?php/** * Created by PhpStorm. * User: 工作 * Date: 2018/1/11 * Time: 12:02 */ //连接数据库$dsn = " ...

  7. http的长连接与端连接

    长连接的定义:  HTTP1.1规定了默认保持长连接(HTTP persistent connection ,也有翻译为持久连接),数据传输完成了保持TCP连接不断开(不发RST包.不四次握手),等待 ...

  8. 2019-2020-1 20199328《Linux内核原理与分析》第六周作业

    使用gdb跟踪分析一个系统调用内核函数 首先我们删除本身的menu目录,并从github上克隆一个menu,并进行编译 编译过程 现在找到test.c文件,加入上个实验中做的getPid()方法 利用 ...

  9. hdu_2124 Flying to the Mars & hdu_1800 Repair the Wall 贪心水题

    hdu_1800 简单排一下序,从大开始把比他小的都访问一遍,ans++: #include <iostream> #include <stdio.h> #include &l ...

  10. SaltStack的数据系统

    一.SaltStack数据系统的分类 1.Grains(收集的是静态数据) 官方使用文档:https://www.unixhot.com/docs/saltstack/topics/targeting ...