司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队。一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用"P"表示),如下图。在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队);一支炮兵部队在地图上的攻击范围如图中黑色区域所示:
如果在地图中的灰色所标识的平原上部署一支炮兵部队,则图中的黑色的网格表示它能够攻击到的区域:沿横向左右各两格,沿纵向上下各两格。图上其它白色网格均攻击不到。从图上可见炮兵的攻击范围不受地形的影响。
现在,将军们规划如何部署炮兵部队,在防止误伤的前提下(保证任何两支炮兵部队之间不能互相攻击,即任何一支炮兵部队都不在其他支炮兵部队的攻击范围内),在整个地图区域内最多能够摆放多少我军的炮兵部队。Input
第一行包含两个由空格分割开的正整数,分别表示N和M;
接下来的N行,每一行含有连续的M个字符('P'或者'H'),中间没有空格。按顺序表示地图中每一行的数据。N <= 100;M <= 10。Output
仅一行,包含一个整数K,表示最多能摆放的炮兵部队的数量。
Sample Input
5 4 PHPP PPHH PPPP PHPP PHHP
Sample Output
6
#include<stdio.h>
#include<iostream>
#include<algorithm>
#include<string.h>
#include<vector>
#include<cmath>
#include<string>
#include<map>
#include<queue>
using namespace std;
typedef long long ll;
ll row,col;
ll nums;
ll base[110];
ll state[70];
ll soldier[70];
ll dp[110][70][70];
int main(){
memset(base,0,sizeof(base));
memset(state,0,sizeof(state));
memset(soldier,0,sizeof(soldier));
memset(dp,0,sizeof(dp));
nums=0;
cin>>row>>col;
string s;
for(ll i=0;i<row;i++){
cin>>s;
for(ll j=0;j<col;j++){
if(s[j]=='H'){
base[i]+=(1<<j);
}
}
}
for(ll i=0;i<(1<<col);i++){
if((i&(i<<1))||(i&(i<<2)))
continue;
ll k=i;
while(k){
soldier[nums]+=(k&1);
k>>=1;
}
state[nums++]=i;
}
for(ll i=0;i<nums;i++){
if(state[i]&base[0])
continue;
dp[0][i][0]=soldier[i];
}
for(ll r=1;r<row;r++){
for(ll i=0;i<nums;i++){
if(state[i]&base[r])
continue;
for(ll j=0;j<nums;j++){
if(state[j]&base[r-1])continue;
if(state[i]&state[j])continue;
for(ll k=0;k<nums;k++){
if(state[k]&base[r-2])continue;
if(state[j]&state[k]||state[i]&state[k])
continue;
dp[r][i][j]=max(dp[r][i][j],dp[r-1][j][k]+soldier[i]);
}
}
}
}
ll ans=0;
for(ll i=0;i<nums;i++){
for(ll j=0;j<nums;j++){
ans=max(ans,dp[row-1][i][j]);
}
}
cout<<ans<<endl;
return 0;
}