POJ 1185:炮兵阵地
炮兵阵地
| Time Limit: 2000MS | Memory Limit: 65536K | |
| Total Submissions: 21366 | Accepted: 8280 |
Description
司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队。一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用"P"表示),如下图。在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队);一支炮兵部队在地图上的攻击范围如图中黑色区域所示:
![]()
如果在地图中的灰色所标识的平原上部署一支炮兵部队,则图中的黑色的网格表示它能够攻击到的区域:沿横向左右各两格,沿纵向上下各两格。图上其它白色网格均攻击不到。从图上可见炮兵的攻击范围不受地形的影响。
现在,将军们规划如何部署炮兵部队,在防止误伤的前提下(保证任何两支炮兵部队之间不能互相攻击,即任何一支炮兵部队都不在其他支炮兵部队的攻击范围内),在整个地图区域内最多能够摆放多少我军的炮兵部队。

如果在地图中的灰色所标识的平原上部署一支炮兵部队,则图中的黑色的网格表示它能够攻击到的区域:沿横向左右各两格,沿纵向上下各两格。图上其它白色网格均攻击不到。从图上可见炮兵的攻击范围不受地形的影响。
现在,将军们规划如何部署炮兵部队,在防止误伤的前提下(保证任何两支炮兵部队之间不能互相攻击,即任何一支炮兵部队都不在其他支炮兵部队的攻击范围内),在整个地图区域内最多能够摆放多少我军的炮兵部队。
Input
第一行包含两个由空格分割开的正整数,分别表示N和M;
接下来的N行,每一行含有连续的M个字符('P'或者'H'),中间没有空格。按顺序表示地图中每一行的数据。N <= 100;M <= 10。
接下来的N行,每一行含有连续的M个字符('P'或者'H'),中间没有空格。按顺序表示地图中每一行的数据。N <= 100;M <= 10。
Output
仅一行,包含一个整数K,表示最多能摆放的炮兵部队的数量。
Sample Input
5 4 PHPP PPHH PPPP PHPP PHHP
Sample Output
6
这道状态DP折磨了我相当久的时间,做的时候又花了一个下午找WA原因在哪。
首先,自己做麻烦的第一点在于 a&b 两个十进制的数本身就可以判断两列是否有炮兵对打,而不用像我一开始想象的把十进制转成二进制,再用二进制进行判断,这个已经帮你弄好了。
第二点,本身状态也没有那么多。最多是1024个,再加上用if((i&i<<1)||(i&i<<2)) (这个判断简直了)之后就剩了大概六十个左右好像是。所以将这些合理的状态记录下来,这样的思路就能够想到了。
第三点,判断状态 哪一行是属于第几个状态,这点也很好。
剩下的就是状态dp自己的事了,什么这一行的值只和前两行有关什么的。
总之,这道题真的很好,很值得琢磨。
代码:
#include <iostream>
#include <vector>
#include <string>
#include <cstring>
#include <algorithm>
#include <cmath>
#pragma warning(disable:4996)
using namespace std;
int hang[110];//hang[i]表示第i行的初始状态
int state[80];//
int solider[80];
int dp[110][80][80];
int row,col,i,j,k,h,he;
char pb_state[15];
int main()
{
cin>>row>>col;
memset(hang,0,sizeof(hang));
memset(dp,0,sizeof(dp));
memset(solider,0,sizeof(solider));
memset(state,0,sizeof(state));
for(i=1;i<=row;i++)
{
cin>>pb_state;
for(j=0; j<col; j++)
if(pb_state[j]=='H') hang[i]+=1<<j;
}
he=1;
for(i=0;i<(1<<col);i++)
{
if((i&i<<1)||(i&i<<2)) continue;
state[he]=i;
int temp=i,num=0;
while(temp)
{
num += temp&1;
temp=temp>>1;
}
solider[he]=num;
he++;
}
for(i=1;i<he;i++)
{
if(state[i]&hang[1])continue;
dp[1][i][0]= solider[i];
}
for(i=1;i<he;i++)
{
if(state[i]&hang[2])continue;
for(j=1;j<he;j++)
{
if(state[i]&state[j])continue;
if(state[j]&hang[1]) continue;
dp[2][i][j]=max(dp[2][i][j],dp[1][j][0]+solider[i]);
}
}
for(h=3;h<=row;h++)
{
for(i=1;i<he;i++)//
{
if(state[i]&hang[h])continue;
for(j=1;j<he;j++)
{
if(state[j]&hang[h-1])continue;
if(state[j]&state[i])continue;
for(k=1;k<he;k++)
{
if(state[k]&hang[h-2])continue;
if(state[k]&state[i])continue;
if(state[k]&state[j])continue;
dp[h][i][j]=max(dp[h][i][j],dp[h-1][j][k]+solider[i]);
}
}
}
}
int ans=0;
for(i=1;i<he;i++)
for(j=0;j<he;j++)//!!!注意啊,这里一定是从0开始的,因为之前设置好的就是0啊。。。。。。。。。。。。。。。。。。
ans=max(ans,dp[row][i][j]);
cout<<ans<<endl;
//system("pause");
return 0;
}版权声明:本文为博主原创文章,未经博主允许不得转载。
浙公网安备 33010602011771号