状压入门--bzoj1087: [SCOI2005]互不侵犯King【状压dp】
Description
在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案。国王能攻击到它上下左右,以及左上
左下右上右下八个方向上附近的各一个格子,共8个格子。
Input
只有一行,包含两个数N,K ( 1 <=N <=9, 0 <= K <= N * N)
Output
方案数。
Sample Input
Sample Output
void getnowk(int x)
{
int tmp=;
while(x) tmp+=(x&),x>>=;
nowk[cnt]=tmp;
} void pre()
{
FAKE=(<<n)-;//可能的状态总数(最大值)
for(int i=;i<FAKE;i++)
if(!(i&(i>>))) bin[++cnt]=i,getnowk(i);
}
预处理状态部分
现在我们来说状态!这种棋盘型的状压dp,我们一般把“行”作为状态。
设f[i][j][k]为在前i行,状态序号为j,目前已经放了k个国王的方案。
转移:f[i[[j][k]=sigma( f[i-1][x][k-nowk[j] ) 其中x枚举的是上一行的状态,nowk记录着此状态中放了多少国王,这里减的是当前行状态的国王数。
另外,除了同行互不侵犯,邻行是否互不侵犯我们如何处理?
无敌位运算!
if(bin[j]&bin[k]) continue; //上下可侵犯
if((bin[j]<<)&bin[k]) continue; // 左上右下侵犯
if(bin[j]&(bin[k]<<)) continue; // 右上坐下侵犯
于是,这就是状压dp。它把复杂的状态转化为一个数,用位运算处理状态!
code
#include<cstdio>
#include<algorithm>
//写题解的时候可以输出一下中间结果!
using namespace std;
typedef long long ll; int n,suming,cnt,FAKE;
int bin[],nowk[];//bin[]: 01串表示的十进制
//nowk[]: 此时放的国王数量 即01串里有多少个1
ll f[][][];
ll ans; void getnowk(int x)
{
int tmp=;
while(x) tmp+=(x&),x>>=;
nowk[cnt]=tmp;
} void pre()
{
FAKE=(<<n)-;//可能的状态总数(最大值)
for(int i=;i<FAKE;i++)
if(!(i&(i>>))) bin[++cnt]=i,getnowk(i);
} int main()
{
scanf("%d%d",&n,&suming);
pre();//预处理出的状态 cnt是状态总数
//所有的状态:相邻无1的01串
for(int i=;i<=cnt;i++) f[][i][nowk[i]]=;
//赋初值,第1行第i个状态一定只有1种方案。
/* for(int i=1;i<=cnt;i++) printf("%d ",bin[i]);
printf("\n");
for(int i=1;i<=cnt;i++) printf("%d ",nowk[i]);*/
for(int i=;i<=n;i++) //第1行已处理出,从第2行开始
for(int j=;j<=cnt;j++)//枚举当前行(第i行)的状态
for(int k=;k<=cnt;k++)//枚举上一行(第k行)的状态
{
if(bin[j]&bin[k]) continue;
if((bin[j]<<)&bin[k]) continue;
if(bin[j]&(bin[k]<<)) continue;
for(int s=suming;s>=nowk[j];s--) f[i][j][s]+=f[i-][k][s-nowk[j]];
}
for(int i=;i<=cnt;i++) ans+=f[n][i][suming];
printf("%lld",ans);
return ;
}
感谢 @KesdiaelKen @p_b_p_b 提供的思路!
*Update
用状压dp能否解决八皇后问题? 答案是否定的,在八皇后问题中要满足行、列、对角线上除了本身没有另外的皇后,因为行、列、对角线是棋盘里整条贯穿边界的线,所以用位运算压缩状态就不太行。
所以状压dp比较适合上下左右仅相邻的情况 ,这种整行的维护就比较困难。
另如https://www.luogu.org/problemnew/show/P2051 ,若把n或m中的一个范围改为10,能不能用状压dp呢?答案依然是否定的,不行的原因与八皇后问题相似。
状压入门--bzoj1087: [SCOI2005]互不侵犯King【状压dp】的更多相关文章
- [BZOJ1087] [SCOI2005] 互不侵犯King (状压dp)
Description 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格子. Input 只有一行,包 ...
- BZOJ1087 SCOI2005 互不侵犯King 【状压DP】
BZOJ1087 SCOI2005 互不侵犯King Description 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附 ...
- BZOJ 1087: [SCOI2005]互不侵犯King [状压DP]
1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3336 Solved: 1936[Submit][ ...
- bzoj 1087 [SCOI2005]互不侵犯King 状态压缩dp
1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec Memory Limit: 162 MB[Submit][Status][Discuss] Descripti ...
- [bzoj1087][scoi2005]互不侵犯king
题目大意 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上 左下右上右下八个方向上附近的各一个格子,共8个格子. 思路 首先,搜索可以放弃,因为这是一 ...
- bzoj1087: [SCOI2005]互不侵犯King (codevs2451) 状压dp
唔...今天学了状压就练练手... 点我看题 这题的话,我感觉算是入门题了QAQ... 然而我还是想了好久... 大致自己推出了方程,但是一直挂,调了很久选择了题解 坚持不懈的努力的调代码. 然后发现 ...
- [BZOJ1087][SCOI2005]互不侵犯King解题报告|状压DP
在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格子. 好像若干月前非常Naive地去写过DFS... ...
- 【BZOJ1087】 [SCOI2005]互不侵犯King 状压DP
经典状压DP. f[i][j][k]=sum(f[i-1][j-cnt[k]][k]); cnt[i]放置情况为i时的国王数量 前I行放置情况为k时国王数量为J #include <iostre ...
- BZOJ 1087 [SCOI2005]互不侵犯King ——状压DP
[题目分析] 沉迷水题,吃枣药丸. [代码] #include <cstdio> #include <cstring> #include <iostream> #i ...
随机推荐
- hdu6196 happpy happy happy (meet in middle + 剪枝)
题意 从1到n共计n(<=90)个物品,每个物品有一个价值a[i],儿子和爸爸轮流做游戏,儿子先手.儿子每次选价值最大的{最左边,最右边}的物品,如果价值一样大, 则选取最左边的物品. 爸爸每次 ...
- OO第三单元总结--根据JML写代码
一. JML语言 1. 理论基础 首先,JML不是JAVA的一部分,它是一群研究者为JAVA设计的扩展部分,但还没有得到官方的支持.因此,JAVA编译器并不支持JML,所以要想JML起作用,只能采用类 ...
- Markdown中插入图片技巧收集
在操作Markdown时图片应该是最头痛的一件事! 比如要发送一个md文件给对方,如果附带了图片时,那么就要一大堆文件包括图片发给对方等等,如果使用在线图片,那么这个服务器又是一大痛点,因为你不确定这 ...
- Codeforces div.2 B. The Child and Set
题目例如以下: B. The Child and Set time limit per test 1 second memory limit per test 256 megabytes input ...
- SQL 撤销索引、撤销表以及撤销数据库
SQL 撤销索引.撤销表以及撤销数据库 通过使用 DROP 语句,可以轻松地删除索引.表和数据库. DROP INDEX 语句 DROP INDEX 语句用于删除表中的索引. 用于 MS Access ...
- java notify notifyAll
notify()和notifyAll()都是Object对象用于通知处在等待该对象的线程的方法. void notify(): 唤醒一个正在等待该对象的线程.void notifyAll(): 唤醒所 ...
- LoadRunner系列之—-04 录制基于https协议的脚本
实际性能测试过程中,有些需录制脚本的页面或接口是基于https协议的,按原来方法录制脚本,录完了脚本是空的.为解决这个问题,第一步了解https协议的具体实现,这块网上资料很多,可参考页面下方参考资料 ...
- 一起talk C栗子吧(第一百二十四回:C语言实例--内置宏)
各位看官们,大家好,上一回中咱们说的是显示变量和函数地址的样例,这一回咱们说的样例是:内置宏.闲话休提,言归正转.让我们一起talk C栗子吧! 看官们,我们在编译程序的时候,假设有语法错误,编译器就 ...
- 2012年公司组织旅游西安线个人记录(repost)
2012年公司组织旅游西安线个人记录 文件夹 [隐藏] 1 序言 2 第1天 3 第2天 4 第3天 5 第4天 6 第5天 [title=2012%E5%B9%B4%E5%85%AC%E5%8F% ...
- ufldl学习笔记与编程作业:Logistic Regression(逻辑回归)
ufldl学习笔记与编程作业:Logistic Regression(逻辑回归) ufldl出了新教程,感觉比之前的好,从基础讲起.系统清晰,又有编程实践. 在deep learning高质量群里面听 ...