洛谷 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. 201871010126 王亚涛《面向对象程序设计 JAVA》 第十三周学习总结

      内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这个作业的要求在哪里 https://www.cnblogs.com/nwnu-daizh/p/ ...

  2. python的路径问题

    ## 文件路径出错问题 """ 如何获取与当前文件相关的地址 """ import os # 当前文件的完整路径 print(__file_ ...

  3. 数据库导出--Oracle-dmp格式

    expdp 数据库名/数据库密码@orcl directory=backdir dumpfile=导出文件名称.dmp 例: expdp bedManager_nt/123456@orcl direc ...

  4. 第四组项目总结(UML图设计)

    第四组项目总结(UML图设计) 相关链接: 墨刀原型链接:https://pan.baidu.com/s/1qrVI_je8NONVHT_FwH6Pwg 需求文档链接:https://www.cnbl ...

  5. golang数据结构之稀疏数组

    掌握知识: 数组的初始化和赋值 结构体的初始化和赋值 字符串和整型之间的转换以及其它的一些操作 类型断言 读取文件 写入文件 对稀疏数组进行压缩 package main import ( " ...

  6. HDU 6556 (2018CCPC吉林 B题)

    ### HDU 6556 题目链接 ### 题目大意: 给你四个国家的时区,告诉你 A 国家的时间,让你输出这时候在 B 国家的时间,还需要输出对于 A 国家来说这是 昨天.今天 还是 明天. 分析前 ...

  7. 邮件hMailServer +Foxmail 安装使用教程

    hMialServer是Windows下一款免费开源的邮件服务器软件,支持smtp.pop3.imap. 本文主要根据官方文档Quick-Start guide整理而成. 一.下载 下载地址:http ...

  8. netty ByteBuf与String相互转换

    String转为ByteBuf 1)使用String.getBytes(Charset),将String转为byte[]类型 2)使用Unpooled.wrappedBuffer(byte[]),将b ...

  9. Redis for OPS 07:Redis 补充说明

    写在前面的话 redis 的各种架构搭建暂时就到这里,本文主要用于补充说明 Redis 的一些概念以及配置文件的相关信息. 常用词汇 缓存穿透: 类似热点数据存储 Redis 一样,对于非热点数据存储 ...

  10. 关于window10更新之后,15.5版本虚拟机不能使用的情况:检测更新版本

    1.用了四五年的虚拟机,最近居然老提示检测更新版本,嗯,我将我的虚拟机由10版本,更新到了15.5版本,这也是网友说的,然而并灭有什么乱弄.window10系统自动更新,自动更新以后虚拟机就打不开了, ...