二维状压DP经典题
炮兵阵地
题目大意:在n*m的地图上放置炮兵,每个炮兵的攻击范围是上下左右两格内,有两种不同的地形,山地(用“H” 表示),平原(用“P”表示),只有平原可以布置炮兵,在不冲突的前提下最多可以布置多少炮兵?
这道题非常经典,我们用dp[i] [j] [k]表示第i行在第j种选取状态下,第i-1行在第k种选取状态下前i行最多摆放的炮兵数量。然后我们首先预处理每一行所有的合法状态,以降低时间复杂度。用num[i]表示第i行的合法状态数量,state[i] [j]表示第i行的第j种合法状态是什么,用temp[i]存储第i行的初始状态,用c[i]存储每种合法状态对应的炮兵数量。
可以不处理直接枚举所有状态吗
如果不预先处理合法状态,那么每一行所有的状态可能有2^10 = 1024种,由于炮兵的摆放需要考虑前两行的状态,那么三个循环枚举状态就会达到惊人的时间复杂度,所以预处理很关键!进行预处理我们会发现每一行的合法状态最多60种,这样即使是3个循环复杂度也会很低。
其他细节都在代码里了。
#include <bits/stdc++.h>
using namespace std;
int n,m,ans;
int dp[105][65][65];
int num[105],temp[105];
int state[105][65];
int c[1 << 10 + 5];
int count(int x){
int sum = 0;
while(x){
if(x & 1) sum++;
x >>= 1;
}
return sum;
}
int main(){
cin >> n >> m;
//每行的合法状态最多只有60种!
// int p = 0;
// for(int i = 0;i< (1 << 10);i++){
// int now = i;
// if((now & (now >> 1)) == 0 && (now & (now >> 2)) == 0) p++;
// }
// cout << p << endl;
for(int i = 1;i <= n;i++){
string s;
cin >> s;
for(int j = 0;j < m;j++){
if(s[j] == 'P') temp[i] += (1 << j);
}
}
//对第0行特殊处理
state[0][++num[0]] = 0;
//预处理合法状态
for(int i = 0;i < (1 << m);i++){
for(int j = 1;j <= n;j++){
int now = i;
if(!(now & (now >> 1)) && !(now & (now >> 2))
&& (now | temp[j]) == temp[j]){
state[j][++num[j]] = i;
c[i] = count(i);
}
}
}
for(int i = 1;i <= n;i++){
for(int j = 1;j <= num[i];j++){
int now = state[i][j];
//对第一行特殊处理,防止越界
if(i == 1) {
dp[i][j][1] = max(dp[i][j][1],c[now]);
continue;
}
for(int k = 1;k <= num[i - 1];k++){
int pre = state[i - 1][k];
if(!(now & pre)){
for(int l = 1;l <= num[i - 2];l++){
int pree = state[i - 2][l];
if(!(now & pree) && !(pre & pree)){
dp[i][j][k] = max(dp[i][j][k],dp[i - 1][k][l] + c[now]);
}
}
}
}
}
}
for(int i = 1;i <= num[n];i++){
for(int j = 1;j <= num[n - 1];j++){
ans = max(ans,dp[n][i][j]);
}
}
cout << ans << endl;
return 0;
}
排兵布阵 HDU4539
大意:一个n*m的平原布置士兵。每个士兵可以攻击到并且只能攻击到与之曼哈顿距离为2的位置以及士兵本身所在的位置。当然,一个士兵不能站在另外一个士兵所能攻击到的位置,同时因为地形的原因也不是每一个位置都可以安排士兵。 (输入中1可以布置,0不可以布置)问:最多能安排多少个士兵。
思路和上一题基本一样,换汤不换药.
#include<bits/stdc++.h>
using namespace std;
int n,m,ans;
int state[105],num[170],c[1 << 10 + 5],legal[105][170],dp[105][170][170];
int count1(int x){
int sum = 0;
while(x){
if(x & 1) sum++;
x >>= 1;
}
return sum;
}
bool ok(int now,int row){
return (now & (now >> 2)) == 0 && (now | state[row]) == state[row];
}
bool check(int now,int pre){
return (now & (pre >> 1)) == 0 && (now & (pre << 1)) == 0;
}
bool recheck(int now,int pree){
return (now & pree) == 0;
}
//由于是多组数据,所以每次都要初始化!
void init(){
ans = 0;
memset(num,0,sizeof(num));
memset(dp,0,sizeof(dp));
memset(state,0,sizeof(state));
}
int main()
{
memset(c,-1,sizeof(c));
while(~scanf("%d %d",&n,&m)){
init();
for(int i = 1;i <= n;i++){
for(int j = 0;j < m;j++){
int x;
scanf("%d",&x);
if(x) state[i] |= (1 << j);
}
}
//预处理合法状态
legal[0][++num[0]] = 0;
for(int i = 0;i < (1 << m);i++){
for(int j = 1;j <= n;j++){
if(ok(i,j)) {
legal[j][++num[j]] = i;
if(c[i] == -1) c[i] = count1(i);
}
}
}
for(int i = 1;i <= n;i++){
for(int j = 1;j <= num[i];j++){
int s1 = legal[i][j];
if(i == 1) {
dp[i][j][1] = max(dp[i][j][1],c[s1]);
continue;
}
for(int k = 1;k <= num[i - 1];k++){
int s2 = legal[i - 1][k];
if(check(s1,s2)){
for(int l = 1;l <= num[i - 2];l++){
int s3 = legal[i - 2][l];
if(recheck(s1,s3)){
dp[i][j][k] = max(dp[i][j][k],dp[i-1][k][l] + c[s1]);
}
}
}
}
}
}
for(int i = 1;i <= num[n];i++){
for(int j = 1;j <= num[n - 1];j++){
ans = max(ans,dp[n][i][j]);
}
}
printf("%d\n",ans);
}
return 0;
}
二维状压DP经典题的更多相关文章
- HihoCoder - 1048 状压DP 经典题
hihocoder题解说的十分清晰了,这份代码就是从讲解里学习的 方案数就是不断枚举合法状态下横放竖放或两者均可 合法判断的依据是记录当前行和下一行的状态 防止重复枚举的方法是先按行后按列 递归基瞎写 ...
- 二:状压dp
一:状压dp的基本特征 状态压缩问题一般是指用十进制的数来表示二进制下的状态 这种用一个数来表示一组数,以降低表示状态所需的维数的解题手段,就叫做状态压缩. 常用到位运算 二:位运算 &:与运 ...
- 【bzoj3195】【 [Jxoi2012]奇怪的道路】另类压缩的状压dp好题
(上不了p站我要死了) 啊啊,其实想清楚了还是挺简单的. Description 小宇从历史书上了解到一个古老的文明.这个文明在各个方面高度发达,交通方面也不例外.考古学家已经知道,这个文明在全盛时期 ...
- 【bzoj1087】【互不侵犯King】状压dp裸题(浅尝ACM-D)
[pixiv] https://www.pixiv.net/member_illust.php?mode=medium&illust_id=54329606 向大(hei)佬(e)势力学(di ...
- [NOI2001] 炮兵阵地 (状压Dp经典例题)
如果您的电脑比较优秀能在 1sec 内跑过 2^1000 的时间复杂度,不妨你可以尝试一下,其实实际时间复杂度远远少于 2^1000,作为骗分不错的选择QAQ,然后我们来分析一下正解: 很显然此题是一 ...
- 7月15日考试 题解(链表+状压DP+思维题)
前言:蒟蒻太弱了,全打的暴力QAQ. --------------------- T1 小Z的求和 题目大意:求$\sum\limits_{i=1}^n \sum\limits_{j=i}^n kth ...
- POJ 3254 - Corn Fields - [状压DP水题]
题目链接:http://poj.org/problem?id=3254 Time Limit: 2000MS Memory Limit: 65536K Description Farmer John ...
- hdu 1185 状压dp 好题 (当前状态与上两行有关系)
/* 状压dp 刚开始&写成&&看了好长时间T0T. 状态转移方程 dp[i][k][j]=Max(dp[i][k][j],dp[i-1][l][k]+num[i][j]);( ...
- POJ 2411 状压DP经典
Mondriaan's Dream Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 16771 Accepted: 968 ...
随机推荐
- re匹配 [\s\S][\w\W]的使用.
本来想提取一个字符串写了一堆正则都提取不出来. 因为有特殊字符 后来使用 [\s\S]* 或 [\w\W]* 匹配出来. \s 空白字符 [ \t\n\r\f\v] \S 非空白字符 相当于 [^ \ ...
- c++问题集合
1.对于程序未运行和运行后的代码段到底存储什么? 2.编写程序时为什么先申请变量后使用? 3.malloc本质到底分配了什么?谁赋给的? 4.程序在系统上是怎么运行起来的? 5.当我们双击一个程序运行 ...
- WebGL的shader
WebGL的shader(着色器)有2种:vertexShader(定点着色器)和 fragmentShader(片段着色器) 顶点着色器:定义点的位置.大小 片元着色器:定义画出来的物体的材质(颜色 ...
- xml详解
https://www.cnblogs.com/zhao1949/p/5652167.html https://www.cnblogs.com/cb0327/p/4967782.html
- BZOJ 4732 UOJ #268 [清华集训2016]数据交互 (树链剖分、线段树)
题目链接 (BZOJ) https://www.lydsy.com/JudgeOnline/problem.php?id=4732 (UOJ) http://uoj.ac/problem/268 题解 ...
- vue draggable 火狐拖拽搜索问题
最近在使用vuedraggable做导航时候,谷歌拖拽是没问题的,但是在火狐测试时候,拖拽时候是可以成功,但是火狐还是打开了一个新的tab,并且搜索了,一开始想着是阻止默认行为,但是在@end时间中阻 ...
- Java微信公众号开发梳理
Java微信公众号开发梳理 现在微信公众平台的开发已经越来越普遍,这次开发需要用到微信公众平台.因此做一个简单的记录,也算是给那些没踩过坑的童鞋一些启示吧.我将分几块来简单的描述一下,之后会做详细的说 ...
- win2008 r2下配置IIS7(ASP.net运行环境)
一.常规配置: 1.先要设置应用程序池(ApplicationPool)为Classic.NETAppPool. 2.选中左侧的“应用程序池”,选中“Classic.NETAppPool”,单击右侧的 ...
- Delphi实现树型结构具体实例
unit Unit1;interfaceuses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, ...
- 自然语言处理(NLP)之个人小结
一 概述 1.1 自然语言处理四大任务 序列标注 分词 词性标注 命名实体识别 分类任务 文本分类 情感分析 判断句子关系 问答系统 对话系统 阅读理解 生成任务 机器翻译 自动文摘 图像描述生成 1 ...