hdu 2167 方格取数 【状压dp】(经典)
<题目链接>
题目大意:
给出一些数字组成的n*n阶矩阵,这些数字都在[10,99]内,并且这个矩阵的 3<=n<=15,从这个矩阵中随机取出一些数字,在取完某个数字后,该数字周围8个点都不能取,问:取得数字的最大和为多少?
解题分析:
由于对每一个数,有选和不选两种可能,分别对应状态压缩中的1和0,且 n<=15,1<<15不是非常大,因此就可以非常自然的想到状态压缩。
此题要与普通的状压dp不同的是,当某一行取某种方案时,如何求出这种取数的所有取得的数之和,就是下面的bit数组,还有要注意一下输入。
#include<cstdio> //此题要掌握的是,当某一行取某种方案时,如何求出这种取数的所有取得的数之和,就是下面的bit数组
#include<iostream> //还要注意如何读入的格式,挺难的
#include<cstring>
#include <algorithm>
#define N (1<<15)+10
int dp[][N];//dp表示第i行j方案时获得的最大值
int a[][], t[N], cnt, n, bit[N];//cnt存合法方案总数
using namespace std; void init() //此题要掌握的技巧是,当某一行取某种方案时,如何求出这种取数的所有取得的数之和,就是下面的bit数组
{
bit[] = ;
for (int i = ; i <= n; i++) //存下2的1~15次方,与之后的状态比较
bit[i] = bit[i - ] << ; //bit[2]为2的1次方,bit[3]为2的二次方
} void solve()
{
int i, j, k, l;
cnt = ; memset(dp, , sizeof(dp));
for (i = ; i< ( << n) - ; i++) //用二进制存方案,每行最多有(1<<n)-1中方案,包括不合法方案
{
if (((i << )&i) == ) //判断方案i是否相邻两格均为1
{
for (j = ; j <= n; j++)
if ((bit[j] & i)) //这个技巧一定要掌握,非常妙***
dp[][i] += a[][j]; //先处理第一行的所有方案 dp[1][i]表示第一行取第i种方案时,在第一行得到的所有数的总数
t[++cnt] = i; //这里的t[]数组也同时表示每一行的合法情况(只用一行中选取的数不相邻这一条件来约束时)
}
} //初始化dp数组,为下面的状态转移方程做好递推准备 for (i = ; i <= n; i++)
{
for (j = ; j <= cnt; j++)
{
int posn = ;
for (k = ; k <= n; k++)
if (bit[k] & t[j])
posn += a[i][k];
for (l = ; l <= cnt; l++) //这里的 t[j]&t[l]就是用来判断第i行取j方案时,是否与它上一行取的数,有相邻的,如果有,则这种方案舍弃
if ((t[j] & t[l]) == && ((t[l] << )&t[j]) == && (t[l] >> & t[j]) == ) // t[l]<<1 & t[j] 和 t[l]>>1&t[j] 则是分别判断第i行取的数是否与它右上和左上的数相邻,如果相邻,也舍弃
dp[i][t[j]] = max(dp[i][t[j]], posn + dp[i - ][t[l]]);
} //dp[i][t[j]]表示当第i行选第j种方案时,前i行能取的数的最大总和
}
} int main()//状态压缩dp,状态最多有(2<<15)-1种
{
int i, j;
char str[];
while (gets_s(str))
{
memset(a, , sizeof(a));
n = ;
for (i = ; i < strlen(str); i += ) {
a[][++n] = (str[i] - '') * + (str[i + ] - '');
}
for (i = ; i <= n; i++)
{
gets_s(str);
int m = ;
for (j = ; j < strlen(str); j += ) {
a[i][m++] = (str[j] - '') * + (str[j + ] - '');
}
}
getchar();
//以上为读入矩阵,这个输入还是要注意 init();
solve();
int res = ;
for (i = ; i <= cnt; i++)
if (res<dp[n][t[i]]) //dp[i][j]表示当第i行选第j种方案时,前i行能取的数的最大总和
res = dp[n][t[i]];
printf("%d\n", res);
}
return ;
}
2018-07-26
hdu 2167 方格取数 【状压dp】(经典)的更多相关文章
- HDU 1565 方格取数  状压dp
		题目: 给你一个n*n的格子的棋盘,每个格子里面有一个非负数. 从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的和最大. Input 包括多 ... 
- 方格取数--状压DP or 网络流
		题意:http://acm.hdu.edu.cn/showproblem.php?pid=1565 取不相邻的点是权值最大. 这题可以网络流做,暂时先DP一下,网络流明天学一下~~ #define I ... 
- HDU 1565 方格取数(1) 轮廓线dp
		题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1565 方格取数(1) Time Limit: 10000/5000 MS (Java/Others) ... 
- HDU 1565 方格取数(1)  ——插头DP
		[题目分析] 其实直接状压就可以了. 但是有点闲,又写了一个可读性极差,智商低下,很(gou)好(pi)的代码 [代码] #include <cstdio> #include <cs ... 
- HDU 1565 - 方格取数(1) - [状压DP][网络流 - 最大点权独立集和最小点权覆盖集]
		题目链接:https://cn.vjudge.net/problem/HDU-1565 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32 ... 
- 特殊方格棋盘【状压DP】
		特殊方格棋盘[状压DP] 讲真状压DP这个东西只不过是有那么亿丢丢考验心态罢了(确信) 先从板子题说起,另加一些基础知识 题目描述 在的方格棋盘上放置n 个车,某些格子不能放,求使它们不能互相攻击的方 ... 
- 网络流(最大流) HDU 1565 方格取数(1) HDU 1569 方格取数(2)
		HDU 1565 方格取数(1) 给你一个n*n的格子的棋盘,每个格子里面有一个非负数.从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的 ... 
- 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)
		方格取数(1) Time Limit: 10000/5000 MS (J ... 
随机推荐
- python - str和repr方法:
			# python 内置__str__()和__repr__()方法: #显示自定制 # 示例1 # a = 123 # print(a.__str__()) # 示例2 class Test(): d ... 
- Django学习手册 - 正则URL路由配置/路由分发
			############################################### 总结: 一.url路由配置: 方式一:(通过url链接get获取) 方式二:(url路由匹配方式获取-拓 ... 
- LaTeX 对齐问题
			一.一行文本对齐 \leftline{左对齐} \centerline{居中} \rightline{右对齐} 二.多行文本或段落对齐 左对齐 \begin{flushleft}...\end{flu ... 
- Python3 日期时间 相关模块(time(时间) / datatime(日期时间) / calendar(日历))
			Python3 日期时间 相关模块(time(时间) / datatime(日期时间) / calendar(日历)) 本文由 Luzhuo 编写,转发请保留该信息. 原文: http://blog. ... 
- 实现开发板与ubuntu的共享--根文件系统NFS--Samba共享【sky原创】
			虚拟机要选择桥接,并且禁用有线和无线网卡,开启本地连接,本地连接属性要写如下: ip地址是在连上板子后,windows cmd 下 ipconfig得出的 板子的网线最好连接交换机或者 ... 
- TYpeScript接口的使用
			1.接口中的属性值的使用: // 作用是强制类型检查 interface Iperson { name: string; age: string; } class Person { construct ... 
- STM32应用实例七:与宇电设备实现AI-BUS通讯
			宇电的设备使用基于RS-485的自定义协议,协议本身比较简单,只有2条指令: 读:地址代号+52H(82) +要读的参数代号+0+0+校验码 写:地址代号+43H(67)+要写的参数代号+写入数低字节 ... 
- OCM_第一天课程:OCM课程环境搭建
			注:本文为原著(其内容来自 腾科教育培训课堂).阅读本文注意事项如下: 1:所有文章的转载请标注本文出处. 2:本文非本人不得用于商业用途.违者将承当相应法律责任. 3:该系列文章目录列表: 一:&l ... 
- Java内存泄漏问题
			1:java中垃圾回收机制主要完成下面两件事情: 跟踪并监控每个java对象,当某个对象处于不可达状态时,回收该对象所占的内存 清理内存分配,回收过程中产生的内存碎片 2:对于JVM的垃圾回收机制来说 ... 
- MySQL CPU 使用率高的原因和解决方法
			用户在使用 MySQL 实例时,会遇到 CPU 使用率过高甚至达到 100% 的情况.本文将介绍造成该状况的常见原因以及解决方法,并通过 CPU 使用率为 100% 的典型场景,来分析引起该状况的原因 ... 
