hdu 1565 方格取数(1)(状态压缩dp)
方格取数(1)
Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768
K (Java/Others)
从中取出若干个数,使得随意的两个数所在的格子没有公共边。就是说所取的数所在的2个格子不能相邻,而且取出的数的和最大。
3
75 15 21
75 15 28
34 70 5
188
分析:dp[i][j]表示前i行,第i行取第j个状态时的取值总和。则dp[i][j] = max(dp[i][j], dp[i-1][k] + sum[i][j]).当中sum[i][j]表示第i行取第j个状态的取值总和。
由于n<=20。经计算能够发现,合法状态最多有17711个。
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; const int N = 21;
const int M = 17720; //合法状态最多有17711个
int n, p;
int a[N][N];
int dp[N][M];
int s[M];
int sum[N][M]; bool checkA(int x) { //推断本行状态是否冲突
return !(x & (x >> 1));
} bool checkB(int x, int y) { //推断本行和上一行是否冲突
return !(x & y);
} int get_sum(int r, int state) { //求第i行状态为state时的取值总和
int res = 0;
for(int i = 0; i < n; i++)
if((state >> i) & 1)
res += a[r][n - 1 - i];
return res;
} void Init() {
p = 0;
memset(sum, 0, sizeof(sum)); for(int i = 0; i < (1 << n); ++i) //求出全部合法状态
if(checkA(i))
s[p++] = i; for(int i = 0; i < n; ++i) { //求第i行取第j个状态时的取值总和
for(int j = 0; j < p; ++j) {
sum[i][j] = get_sum(i, s[j]);
}
}
} void solve() {
memset(dp, 0, sizeof(dp));
for(int i = 0; i < p; i++)
dp[0][i] = sum[0][i]; for(int i = 1; i < n; i++) { //行数
for(int j = 0; j < p; j++) { //本行状态
for(int k = 0; k < p; k++) { //上一行的状态
if(checkB(s[j], s[k])) {
dp[i][j] = max(dp[i][j], dp[i-1][k] + sum[i][j]);
}
}
}
} int ans = dp[n-1][0];
for(int i = 1; i < p; i++)
if(ans < dp[n-1][i])
ans = dp[n-1][i];
printf("%d\n", ans);
} int main() {
while(~scanf("%d", &n)) {
for(int i = 0; i < n; i++)
for(int j = 0; j < n; j++)
scanf("%d", &a[i][j]);
Init();
solve();
}
return 0;
}
hdu 1565 方格取数(1)(状态压缩dp)的更多相关文章
- hdu 1565 方格取数(1) 状态压缩dp
方格取数(1) Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Su ...
- HDU 1565 - 方格取数(1) - [状压DP][网络流 - 最大点权独立集和最小点权覆盖集]
题目链接:https://cn.vjudge.net/problem/HDU-1565 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32 ...
- HDU1565 方格取数(1)(状态压缩dp)
题目链接. 分析: 说这题是状态压缩dp,其实不是,怎么说呢,题目数据太水了,所以就过了.手动输入n=20的情况,超时.正解是网络流,不太会. A这题时有个细节错了,是dp[i][j]还是dp[i][ ...
- 网络流(最大流) HDU 1565 方格取数(1) HDU 1569 方格取数(2)
HDU 1565 方格取数(1) 给你一个n*n的格子的棋盘,每个格子里面有一个非负数.从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的 ...
- HDU 1565 方格取数(1) 轮廓线dp
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1565 方格取数(1) Time Limit: 10000/5000 MS (Java/Others) ...
- hdu 2167 方格取数 【状压dp】(经典)
<题目链接> 题目大意: 给出一些数字组成的n*n阶矩阵,这些数字都在[10,99]内,并且这个矩阵的 3<=n<=15,从这个矩阵中随机取出一些数字,在取完某个数字后,该数 ...
- HDU 1565 方格取数(简单状态压缩DP)
http://acm.hdu.edu.cn/showproblem.php?pid=1565 对于每一个数,取或者不取,用0表示不取,1表示取,那么对于每一行的状态,就可以用一个二进制的数来表示.比如 ...
- HDU 1565 方格取数(1)(最大点权独立集)
http://acm.hdu.edu.cn/showproblem.php?pid=1565 题意: 给你一个n*n的格子的棋盘,每个格子里面有一个非负数. 从中取出若干个数,使得任意的两个数所在的格 ...
- HDU 1565 方格取数 状压dp
题目: 给你一个n*n的格子的棋盘,每个格子里面有一个非负数. 从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的和最大. Input 包括多 ...
随机推荐
- jQuery 收缩展开效果
<!DOCTYPE html> <html lang="en"> <head> <meta http-equiv="Conten ...
- cmanformat - 不是命令啦,是个演示文件
描述 DESCRIPTION cmanformat 是 man pages 格式的演示文件. 由于系统不同会有差异.在 XWindow 下会好些. __________________________ ...
- c++ 数组长度
数组长度求解 sizeof template <class T>int getArrayLen(T &array){ return (sizeof(array) / sizeof( ...
- Js 之图片懒加载插件
一.PC端(lazyload) 1.引入js文件 <script src="//ajax.googleapis.com/ajax/libs/jquery/1.10.2/jquery.m ...
- ARP是如何工作的?
我们知道,当我们在浏览器里面输入网址时,DNS服务器会自动把它解析为IP地址,浏览器实际上查找的是IP地址而不是网址.那么IP地址是如何转换为第二层物理地址(即MAC地址)的呢? 在局域网中,这是通过 ...
- h5开发app,移动端 click 事件响应缓慢的解决方案
造成点击缓慢的原因 从点击屏幕上的元素到触发元素的 click 事件,移动浏览器会有大约 300 毫秒的等待时间.为什么这么设计呢? 因为它想看看你是不是要进行双击(double tap)操作. 使用 ...
- vue全选与反选以及通过使用如何filter删除数据
在vue学习经常遇到的一些基本问题,下面是购物车里面的部分功能,分享给初学者,直接上源码: <!DOCTYPE html><html> <head> <met ...
- [Luogu] P1156 垃圾陷阱
题目描述 卡门――农夫约翰极其珍视的一条Holsteins奶牛――已经落了到“垃圾井”中.“垃圾井”是农夫们扔垃圾的地方,它的深度为D(2≤D≤100)英尺. 卡门想把垃圾堆起来,等到堆得与井同样高时 ...
- group 和 gshadow
group组文件 位置:/etc/group 作用:存放用户的分组信息 使用 /etc/group 命令查看时,得到的数据如下: 分析上图,可以得到以下结果 第1个字段:组名 默认组名与用户名名称一样 ...
- 根据Dockerfile创建hello docker镜像
一.编写hello可执行c文件: 1.安装:gcc glibc glibc-static yum install -y gcc glibc glibc-static 2.编写hello.c:vim h ...