暑假集训Day2 互不侵犯(状压dp)
这又是个状压dp (大型自闭现场)
题目大意:
在N*N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案。国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格子。
输入格式:
只有一行,包含两个数N,K 。
输出格式:
所得的方案数。
算法分析:
1.显然这又是一道状压的题
2.显然一样是用f数组表示方案数
But 这个f数组需要开三维
为什么呢
我们首先分析一下f的转移情况 f的状态与什么有关呢 首先我们很容易知道我们的dp是从上往下一点点递推实现的 而这个国王会攻击到自己的身旁八个位置
所以呢 那f就会对自己的下一行产生影响 而且只会对自己的下一行产生影响 换而言之 f只和自己上一行的状态有关
这样的话我们就会用到两维,一维存储前i行 另一维存储状态 但是看到这个题只用两维是不够的 因为这个题需要放置的国王个数恰好等于K
所以我们将前i行放置的国王个数作为一维
和平时的状压题一样我们将状态作为最后一维 所以关于f[i][j][k]的定义我们有 前i行共放了j个国王而且第i行所放国王的状态为k 的方案数
所以我们就可以写出这样的状态转移方程
int sum(int x){
int cnt = 0;
for(int i = x;i;i-=lowbit(i))cnt++;
return cnt;
}
for(int i = 1;i <= n;++i)
for(int j = 0;j < maxs;++j){//枚举本行状态
if(j & (j<<1))continue;//如果该状态冲突则跳过
for(int k = 0;k < maxs;++k){//枚举上一行状态
if(j & k || j & (k << 1) || j & (k>>1))continue;//如果当前行状态和上一行状态冲突则跳过
for(int s = sum(j);s <= m;++s)//枚举前i行所放的国王个数
f[i][s][j] += f[i-1][s - sum(j)][k];//+=上一行的成立状态数
}
}
这个代码的注释已经很清晰了
但是我们再具体分析一下思想(如果不理解sum函数的去翻暑假集训day2 特殊方格棋盘)
首先我们枚举行数
然后枚举本行的状态
本行状态与自己冲突的情况就是(j & (j<<1))为真 j有两位二进制位同时为1 那么才可能(j&j<<1)为真
举个栗子:j = 01100 那么j<<1就是11000和j取与后并不等于0 所以冲突(即连续两个格子放置了国王,他们互相攻击)
3.然后就要枚举上一行的状态
关于本行与上一行状态冲突的判定具体方法可参见上一条
最后的累加:
划重点: sum(j) 为当前行所放的国王个数,前i行的国王个数肯定就是大于等于这个数的 枚举前i行国王个数,然后减去sum(j)就是前i-1行共放的国王个数
因此f[i-1][s-sum(j)][k] 就是 前i-1行共s-sum(j)个国王并且第i-1行的国王个数为k的方案数
通过这个的累加就是我们的递推过程
代码
#include<bits/stdc++.h>
using namespace std;
long long f[10][100][1000],ans;//f[i][j][k]表示前i行放j个国王并且当前行状态为k的成立方案数
int lowbit(int x){return x & -x;}
int sum(int x){
int cnt = 0;
for(int i = x;i;i-=lowbit(i))cnt++;
return cnt;
}
int main(){
int n,m;scanf("%d%d",&n,&m);
f[0][0][0] = 1;
int maxs = 1<<n;
for(int i = 1;i <= n;++i)
for(int j = 0;j < maxs;++j){//枚举本行状态
if(j & (j<<1))continue;//如果该状态冲突则跳过
for(int k = 0;k < maxs;++k){//枚举上一行状态
if(j & k || j & (k << 1) || j & (k>>1))continue;//如果当前行状态和上一行状态冲突则跳过
for(int s = sum(j);s <= m;++s)//枚举前i-1行所放的国王个数
f[i][s][j] += f[i-1][s - sum(j)][k];//+=上一行的成立状态数
}
}
for(int i = 0;i < 1<<n;++i)ans += f[n][m][i];//ans+=前n行放m个国王并且当前行状态为i
printf("%lld\n",ans);
return 0;
}
点点关注
谢谢观看>)<
暑假集训Day2 互不侵犯(状压dp)的更多相关文章
- BZOJ1087[SCOI2005]互不侵犯——状压DP
题目描述 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格子. 输入 只有一行,包含两个数N,K ( ...
- NOI P1896 互不侵犯 状压DP
题目描述 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格子. 注:数据有加强(2018/4/25) ...
- P1896 [SCOI2005]互不侵犯 状压dp
正解:状压dp 解题报告: 看到是四川省选的时候我心里慌得一批TT然后看到难度之后放下心来觉得大概没有那么难 事实证明我还是too young too simple了QAQ难到爆炸TT我本来还想刚一道 ...
- SCOI2005 互不侵犯 [状压dp]
题目传送门 题目大意:有n*n个格子,你需要放置k个国王使得它们无法互相攻击,每个国王的攻击范围为上下左走,左上右上左下右下,共8个格子,求最多的方法数 看到题目,是不是一下子就想到了玉米田那道题,如 ...
- [SCOI2005]互不侵犯 (状压$dp$)
题目链接 Solution 状压 \(dp\) . \(f[i][j][k]\) 代表前 \(i\) 列中 , 已经安置 \(j\) 位国王,且最后一位状态为 \(k\) . 然后就可以很轻松的转移了 ...
- luogu1896 [SCOI2005]互不侵犯 状压DP
题目大意 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格子.( 1 <=N <=9, 0 ...
- [清华集训2015 Day1]主旋律-[状压dp+容斥]
Description Solution f[i]表示状态i所代表的点构成的强连通图方案数. g[i]表示状态i所代表的的点形成奇数个强连通图的方案数-偶数个强连通图的方案数. g是用来容斥的. 先用 ...
- 暑假集训Day2 状压dp 特殊方格棋盘
首先声明 : 这是个很easy的题 可这和我会做有什么关系 题目大意: 在n*n的方格棋盘上放置n个车,某些格子不能放,求使它们不能互相攻击的方案总数. 注意:同一行或同一列只能有一个车,否则会相互攻 ...
- BZOJ 1087: [SCOI2005]互不侵犯King [状压DP]
1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3336 Solved: 1936[Submit][ ...
随机推荐
- GitHub 热点速览 Vol.22:如何打造超级技术栈
作者:HelloGitHub-小鱼干 摘要:build-your-own-x,无论是新手还是老手,这都是一个指向标.方向有了,剩下就是时间和实践的事情,收集了大量可用于软件和 Web 开发的 Publ ...
- Rocket - tilelink - Monitor
https://mp.weixin.qq.com/s/6e-G5RSQc7Xje7mQj8-Lag 简单介绍Monitor的实现. 1. 基本介绍 用于监控各个channel上的 ...
- AUTOSAR-软件规范文档中的UML
https://mp.weixin.qq.com/s/vm5vWNSpbNIYh25-LjJfYg AUTOSAR软件规范文档中存在两种UML图: Sequence diagrams Config ...
- 震惊!Windows Service服务和定时任务框架quartz之间原来是这种关系……
过场CG: 接到公司领导的文件指示,“小熊”需要在6月底去海外执行一个行动代号为[定时任务]的营救计划,这个计划关系到公司某个项目的生死(数据安全漏洞),作战部拟定两个作战方案: 方案一:使用务定 ...
- Java并发编程 (二) 并发基础
个人博客网:https://wushaopei.github.io/ (你想要这里多有) 一.CPU多级缓存-缓存一致性 1.CPU多级缓存 上图展示的是CPU高级缓存的配置,数据的读取和存 ...
- Java实现 蓝桥杯 算法提高 奥运会开幕式
试题 算法提高 奥运会开幕式 资源限制 时间限制:1.0s 内存限制:256.0MB 问题描述 学校给高一(三)班分配了一个名额,去参加奥运会的开幕式.每个人都争着要去,可是名额只有一个,怎么办?班长 ...
- java实现第七届蓝桥杯打靶
打靶 题目描述 小明参加X星球的打靶比赛. 比赛使用电子感应计分系统.其中有一局,小明得了96分. 这局小明共打了6发子弹,没有脱靶. 但望远镜看过去,只有3个弹孔. 显然,有些子弹准确地穿过了前边的 ...
- 判断IP是否是IPV4
bool isVaildIp(const char *ip) { int dots = 0; /*字符.的个数*/ int setions = 0; /*ip每一部分总和(0-255)*/ if (N ...
- (十二)DVWA全等级SQL Injection(Blind)盲注--SQLMap测试过程解析
一.测试前分析 前文<DVWA全等级SQL Injection(Blind)盲注-手工测试过程解析> 通过手工测试的方式详细分析了SQL Injection(Blind)盲注漏洞的利用过程 ...
- Java编程技术之浅析JVM内存
JVM JVM->Java Virtual Machine:Java虚拟机,是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的. 基本认知: ...