洛谷 P1896 [SCOI2005]互不侵犯

洛谷传送门

题目描述

在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案。国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格子。

注:数据有加强(2018/4/25)

输入格式

只有一行,包含两个数N,K ( 1 <=N <=9, 0 <= K <= N * N)

输出格式

所得的方案数

输入输出样例

输入 #1复制

输出 #1复制

题解:

原谅我一开始看到题还以为是爆搜。。。

其实是一道状态压缩的题目。

蒟蒻自己一个比较大的进步就是把自己状态设置对了...

设置:\(dp[i][j][k]\)为第\(i\)行状态为\(j\)、已经用了\(k\)个国王时的方案数。

状态压缩大体有这么几步:设置状态\(\rightarrow\)考虑转移方式\(\rightarrow\)按转移方式考虑预处理和判断转移条件\(\rightarrow\)开始转移\(\rightarrow\)统计答案。

那么我们设置好状态,开始考虑转移方式:我们发现,若是想从第\(i-1\)行开始转移,转移的条件一是当前和上一次的状态,但是,这些状态的改变必然还会改变国王的个数。也就是说,这数组的两维是有联系的,是自变量和因变量的关系。所以我们因此想到,既然是自变量和因变量的关系,我们莫不如由此构建一个映射,存下来每个状态和每个状态需要的国王人数。这样我们转移的时候就没啥问题了。

如何预处理呢?我们想到,我们需要按行处理状态,每个状态有放国王和不放国王两种选择。因为是预处理,我们是肯定不能用递推和\(DP\)的(你想干啥)

所以我们考虑搜索。

一次搜索可以处理出所有合法的行的方式。

这里插一嘴,因为我们已经把所有合法的行的方式都求出来了,所以我们没必要再把\(dp\)数组的第二维开那么大,构建好映射关系之后,直接用\(cnt\)代替这个二进制状态即可。(因为\(1-cnt\)的每个数都对应着一个数组\(s[i]\)作为状态。)

然后再转移的时候进行判断是否合法就可以。

转移方程:

\[dp[i][j][l]+=dp[i-1][k][l-num[j]]
\]

这里的\(k,j\)分别表示一种状态。

代码:

#include<cstdio>
#define int long long
using namespace std;
int n,K,cnt,ans;
int s[100],num[100];
int dp[10][100][110];
//dp[i][j][k]表示前i-1行放完,第i行状态为j、有k个国王时的方案数
//状态0/1:0:国王攻击不到;1:被国王占领
void dfs(int pos,int st,int tot)
{
if(pos>=n)
{
s[++cnt]=st;
num[cnt]=tot;
return;
}
dfs(pos+1,st,tot);
dfs(pos+2,st+(1<<pos),tot+1);
}
signed main()
{
scanf("%lld%lld",&n,&K);
dfs(0,0,0);
for(int i=1;i<=cnt;i++)
dp[1][i][num[i]]=1;
for(int i=2;i<=n;i++)
for(int j=1;j<=cnt;j++)
for(int k=1;k<=cnt;k++)
{
if(s[j]&s[k])
continue;
else if(s[j]&(s[k]>>1))
continue;
else if(s[j]&(s[k]<<1))
continue;
else
for(int l=num[j];l<=K;l++)
dp[i][j][l]+=dp[i-1][k][l-num[j]];
}
ans=0;
for(int i=0;i<=cnt;i++)
ans+=dp[n][i][K];
printf("%lld",ans);
return 0;
}

SCOI 2005 互不侵犯的更多相关文章

  1. C++之路进阶——codevs2451(互不侵犯)

    2451 互不侵犯 2005年省队选拔赛四川  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 大师 Master       题目描述 Description 在N×N的棋盘里 ...

  2. BZOJ 1087: [SCOI2005]互不侵犯King [状压DP]

    1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3336  Solved: 1936[Submit][ ...

  3. [bzoj1087][scoi2005]互不侵犯king

    题目大意 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上 左下右上右下八个方向上附近的各一个格子,共8个格子. 思路 首先,搜索可以放弃,因为这是一 ...

  4. 【状压DP】bzoj1087 互不侵犯king

    一.题目 Description 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上.下.左.右,以及左上.左下.右上.右下八个方向上附近的各一个格子,共8个格子. I ...

  5. BZOJ-1087 互不侵犯King 状压DP+DFS预处理

    1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec Memory Limit: 162 MB Submit: 2337 Solved: 1366 [Submit][ ...

  6. SCOI2005互不侵犯King

    1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1499  Solved: 872[Submit][S ...

  7. 洛谷1377 M国王 (SCOI2005互不侵犯King)

    洛谷1377 M国王 (SCOI2005互不侵犯King) 本题地址:http://www.luogu.org/problem/show?pid=1377 题目描述 天天都是n皇后,多么无聊啊.我们来 ...

  8. CODEVS 2451 互不侵犯

    2451 互不侵犯 题目描述 Description 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格 ...

  9. 洛谷 P1896 互不侵犯King

    P1896 [SCOI2005]互不侵犯King 题目描述 在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共 ...

随机推荐

  1. openjdk源码下载

    http://hg.openjdk.java.net/jdk8u/jdk8u60/jdk/file/935758609767 browse>zip

  2. (day67)作业

    有以下广告数据(实际数据命名可以略做调整) ad_data = { tv: [ {img: 'img/tv/001.png', title: 'tv1'}, {img: 'img/tv/002.png ...

  3. 【ECNU77】位与数对个数(数位DP)

    点此看题面 大致题意: 求\(\sum_{x=0}^{a-1}\sum_{y=0}^{b-1}[(x\&y)<k]\). 数位\(DP\) 显然数位\(DP\)吧. 我们设\(f_{n, ...

  4. 发布Jar包到中央仓库

    参考流程 https://blog.csdn.net/qq_36838191/article/details/81027586 备份还原 gpg-keys https://blog.rathena.c ...

  5. 1+x证书Web前端开发HTML+CSS专项练习测试题(八)

    1+x证书Web前端开发HTML+CSS专项练习测试题(八) 官方QQ群 1+x 证书 Web 前端开发 HTML+CSS 专项练习测试题(八) http://blog.zh66.club/index ...

  6. ASP.NET(1)

    1.IIS安装问题,先装VS再装IIS,处理程序映射有问题,使用VS自带的控制台输入命令,注册路径 2.开发模式,一般处理程序,使用IO操作读取html文件,使前后端分离 3.post请求和get请求 ...

  7. C#用Call代替CallVirt之后的测试用例

    一. C# 原始代码和直接结果 测试 C# 代码: class Program { static void Main(string[] args) { A c1 = new C(); c1.Foo() ...

  8. Hive_hdfs导入csv文件

    转自:Hive_hdfs csv导入hive demo   1 create csv file.student.csv 4,Rose,M,78,77,76 5,Mike,F,99,98,98 2 pu ...

  9. 在python中实现随机选择

    想从一个序列中随机抽取若干元素,或者想生成几个随机数. random 模块有大量的函数用来产生随机数和随机选择元素.比如,要想从一个序列中随机的抽取一个元素,可以使用random.choice() : ...

  10. Python爬取《冰雪奇缘2》豆瓣影评

    前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者: 刘铨@CCIS Lab PS:如有需要Python学习资料的小伙伴可 ...