<题目链接>

题目大意:

给出一些数字组成的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】(经典)的更多相关文章

  1. HDU 1565 方格取数 状压dp

    题目: 给你一个n*n的格子的棋盘,每个格子里面有一个非负数. 从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的和最大. Input 包括多 ...

  2. 方格取数--状压DP or 网络流

    题意:http://acm.hdu.edu.cn/showproblem.php?pid=1565 取不相邻的点是权值最大. 这题可以网络流做,暂时先DP一下,网络流明天学一下~~ #define I ...

  3. HDU 1565 方格取数(1) 轮廓线dp

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1565 方格取数(1) Time Limit: 10000/5000 MS (Java/Others) ...

  4. HDU 1565 方格取数(1) ——插头DP

    [题目分析] 其实直接状压就可以了. 但是有点闲,又写了一个可读性极差,智商低下,很(gou)好(pi)的代码 [代码] #include <cstdio> #include <cs ...

  5. HDU 1565 - 方格取数(1) - [状压DP][网络流 - 最大点权独立集和最小点权覆盖集]

    题目链接:https://cn.vjudge.net/problem/HDU-1565 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32 ...

  6. 特殊方格棋盘【状压DP】

    特殊方格棋盘[状压DP] 讲真状压DP这个东西只不过是有那么亿丢丢考验心态罢了(确信) 先从板子题说起,另加一些基础知识 题目描述 在的方格棋盘上放置n 个车,某些格子不能放,求使它们不能互相攻击的方 ...

  7. 网络流(最大流) HDU 1565 方格取数(1) HDU 1569 方格取数(2)

      HDU 1565 方格取数(1) 给你一个n*n的格子的棋盘,每个格子里面有一个非负数.从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的 ...

  8. hdu 1565 方格取数(1) 状态压缩dp

    方格取数(1) Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Su ...

  9. hdu 1565 方格取数(1)(状态压缩dp)

    方格取数(1)                                                                 Time Limit: 10000/5000 MS (J ...

随机推荐

  1. 错误RSA host key for [ip address] has changed and you have requested strict checking.

    @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ WARNING: REMOTE HOST IDENTIFICATION HAS ...

  2. postman变量的使用和设置

    之前只是使用postman做接口管理——将各个项目使用到的接口分类管理起来,用的时候手动改参数调用.这次项目连着跑三个接口,需要用到前一个接口的参数,还来回切平台,真的很麻烦,所以就搜了一下有什么便利 ...

  3. B - Finding Palindromes (字典树+manacher)

    题目链接:https://cn.vjudge.net/contest/283743#problem/B 题目大意:给你n个字符串,然后问你将这位n个字符串任意两两组合,然后问你这所有的n*n种情况中, ...

  4. id特性

    每一个HTML元素都可以附带id特性,id特 <!DOCTYPE html> <!-- To change this license header, choose License H ...

  5. DSO windowed optimization 代码 (1)

    这里不想解释怎么 marginalize,什么是 First-Estimates Jacobian (FEJ).这里只看看代码,看看Hessian矩阵是怎么构造出来的. 1 优化流程 整个优化过程,也 ...

  6. sqlplus连接远程数据库

    方式一:简易连接,不用进行网络配置,其实就是tnsname.ora文件 命令:sqlplus 用户名/密码@ip地址[:端口]/service_name [as sysdba] 示例:sqlplus ...

  7. oracle的读写分离实现

    在MySQL作为应用系统的后台数据库时,我们常常见到这样的架构,一拖二.一拖三等等.这是用MySQL的读写分离技术,实现数据的写入和读取分别在不同的库上,提升了数据库服务能力. 同样,在Oracle作 ...

  8. CentOS 6.5自动化运维之基于cobbler服务的自动化安装操作系统详解

    一.Cobbler安装 前提:cobbler由epel源提供,故此需要事先配置指向epel的yum源方可进行类似下面的安装过程. # yum install -y epel-release # yum ...

  9. docker 定时清理none镜像

    =============================================== 2019/3/31_第1次修改                       ccb_warlock == ...

  10. AS 中 Plugin for Gradle 和 Gradle 之间的版本对应关系

    Plugin for Gradle 和 Gradle 之间的版本对应关系 来源:https://developer.android.com/studio/releases/gradle-plugin. ...