4554: [Tjoi2016&Heoi2016]游戏
Time Limit: 20 Sec Memory Limit: 128 MB
Submit: 857 Solved: 506
[Submit][Status][Discuss]Description
在2016年,佳缘姐姐喜欢上了一款游戏,叫做泡泡堂。简单的说,这个游戏就是在一张地图上放上若干个炸弹,看是否能炸到对手,或者躲开对手的炸弹。在玩游戏的过程中,小H想到了这样一个问题:当给定一张地图,在这张地图上最多能放上多少个炸弹能使得任意两个炸弹之间不会互相炸到。炸弹能炸到的范围是该炸弹所在的一行和一列,炸弹的威力可以穿透软石头,但是不能穿透硬石头。给定一张n*m的网格地图:其中*代表空地,炸弹的威力可以穿透,可以在空地上放置一枚炸弹。x代表软石头,炸弹的威力可以穿透,不能在此放置炸弹。#代表硬石头,炸弹的威力是不能穿透的,不能在此放置炸弹。例如:给出1*4的网格地图*xx*,这个地图上最多只能放置一个炸弹。给出另一个1*4的网格地图*x#*,这个地图最多能放置两个炸弹。现在小H任意给出一张n*m的网格地图,问你最多能放置多少炸弹Input
第一行输入两个正整数n,m,n表示地图的行数,m表示地图的列数。1≤n,m≤50。接下来输入n行m列个字符,代表网格地图。*的个数不超过n*m个Output
输出一个整数a,表示最多能放置炸弹的个数
Sample Input
```
4 4
#***
*#**
**#*
xxx#
```Sample Output
5HINT
Source
TJOI2016的题目都是联赛难度的,但我还是做得令人很不满意,说明在基础算法方面有很大欠缺,不能再本末倒置了。
这题是比较经典的二分图模型,对于每个可以放炸弹的格子,x->y连一条边,这样跑最大匹配之后就能保证任意两个炸弹不在同一行或同一列内了。
但是硬石头怎么办呢,很简单直接将行和列拆成两边,新建节点即可。
#include<cstdio>
#include<cstring>
#include<algorithm>
#define mem(a) memset(a,0,sizeof(a))
#define rep(i,l,r) for (int i=l; i<=r; i++)
using namespace std; const int N=;
int n,m,tx,ty,now,ans,cnt;
int to[N],nxt[N],h[N],lnk[N],vis[N],bel[][];
char str[][]; void add(int a,int b){ to[++cnt]=b; nxt[cnt]=h[a]; h[a]=cnt; } int dfs(int x){
for (int i=h[x],k; i; i=nxt[i]) if (!vis[k=to[i]]){
vis[k]=;
if (!lnk[k] || dfs(lnk[k])) { lnk[k]=x; return ; }
}
return ;
} int main(){
freopen("bzoj4554.in","r",stdin);
freopen("bzoj4554.out","w",stdout);
scanf("%d%d",&n,&m); tx=ty=;
rep(i,,n){
scanf("%s",str[i]+);
rep(j,,m) if (str[i][j]=='#') tx++; else bel[i][j]=tx;
if (str[i][m]!='#') tx++;
}
rep(j,,m){
rep(i,,n){
if (str[i][j]=='#') ty++;
if (str[i][j]=='*') add(bel[i][j],ty);
}
if (str[n][j]!='#') ty++;
}
rep(x,,tx) mem(vis),ans+=dfs(x);
printf("%d\n",ans);
return ;
}