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 ...
随机推荐
- POJ 2503 Babelfish (STL)
题目链接 Description You have just moved from Waterloo to a big city. The people here speak an incompreh ...
- win7下出现读不到移动硬盘的解决办法
很多电脑会出现移动硬盘读不到,或者 读到部分盘的情况,那么下面我就为大家来一一解决这些情况: 方法一: 最常见的是硬盘供电不足导致 ,要么换一个硬盘盒子,要么给硬盘一个外加电源即可方法二: ...
- 在全志平台调试博通的wifi驱动(类似ap6212)【转】
转自:http://blog.csdn.net/fenzhi1988/article/details/44809779 调试驱动之前,首先先看看驱动代码,了解代码大致工作流程,再根据硬件配置驱动,比如 ...
- Linux内核驱动之延时 【转】
转自:http://blog.chinaunix.net/uid-24219701-id-3288103.html jiffies 计数器 定时器中断由系统定时硬件以规律地间隔产生; 这个间隔在启动 ...
- windows环境变量PATH顺序的重要性
PATH是路径的意思,PATH环境变量中存放的值,就是一连串的路径.不同的路径之间,用英文的分号间隔开.系统在执行用户命令时,若用户未给出绝对路径,则首先在当前目录下寻找相应的可执行文件.批处理文件等 ...
- 如何设置使eclipse修改代码不重启tomcat
tomcat配置 1.server.xml reloadable="true"<Context docBase="ins" path="/ins ...
- centos7忘记登录密码修改
很多时候我们都会忘记Linux root 用户的口令,下面就教大家如果忘记root口令怎么办 第1步:开机后在内核上按“e”.截图如下 按e以后会进入内核启动页面,如下图 第2步:在linux16这行 ...
- redux最佳实践
对于初学者,redux的理念很难理解.其实redux并不难,redux核心理念就是数据仓库,所有数据操作来源都是明确的.只不过不是直接操作数据,而是交给特定的角色做特定的工作,分工明确. pro ...
- Java中的钩子方法
钩子方法是啥 钩子顾名思义就是用来挂东西的.那么要挂东西必须有个被挂的东西,要不就是铁环.要不就是墙的边沿.所以要能挂住东西必须要有个被勾住的铁环,要一个钩子.那么在java中也是同样的原理,你首先需 ...
- pytest十:用例 a 失败,跳过测试用例 b 和 c 并标记失败 xfail
当用例 a 失败的时候,如果用例 b 和用例 c 都是依赖于第一个用例的结果,那可以直接跳过用例 b 和 c 的测试,直接给他标记失败 xfail用到的场景,登录是第一个用例,登录之后的操作 b 是第 ...