ZOJ 1654 Place the Robots 二分图最大匹配
唉,又是神一样的建模,表示完全想不到。
题意是给你一块地,上面有空地,草地,障碍三种地形,然后让你在上面放机器人,机器人只能放在空地上。机器人会向上下左右四个方向发出攻击,机器人的攻击可以穿过草地但是无法穿过障碍。问你在不会是机器人相互攻击的前提下,最多能放多少个机器人。
我觉得大致的思路应该是这样的,首先会想当然的想到把能够相互攻击到的空地连边,然后求最大独立集,但最大独立集不好求,所以要想办法把它转化成最大匹配问题。
所以要想办法把空地变成边,首先一块空地肯定是有他的横坐标和纵坐标,可以表示成相交的两条在图上的路径的形式。并且有,如果两个空地在同一行或者同一列,如果中间没有障碍物相隔的话,那么肯定只能最多放一个机器人,就把他当成是一块空地来处理。这样我们就能把原来按照行和列进行分割和标号了。然后遍历所有行和列的交点,如果这个交点是空地的话,就建立一条从这一点行的编号到这一点列的编号的双向边,一个二分图就建立起来了。
二分图上的每一个点表示的是行上或者列上的一个区域,这个区域只能放置一个机器人,并且相连的两个点只能放一个机器人,这样就成功转化成求最大匹配的问题了。
另外,要注意这题的输出时非主流的Case :1,冒号在前面,一开始写成 Case 1:狂WA,真是囧
#include <cstdio> #include <cstring> #include <cmath> #include <algorithm> #include <climits> #include <string> #include <iostream> #include <map> #include <cstdlib> #include <list> #include <set> #include <queue> #include <stack> using namespace std; typedef long long LL; const int maxn = 61; const int maxk = maxn * maxn; char buf[maxn][maxn]; int bx[maxk],by[maxk],n,m,cntx,cnty; int xid[maxn][maxn],yid[maxn][maxn]; bool g[maxk][maxk],vis[maxk]; int dfs(int now) { for(int i = 1;i <= cnty;i++) if(g[now][i] && !vis[i]) { vis[i] = true; if(!by[i] || dfs(by[i])) { bx[now] = i; by[i] = now; return 1; } } return 0; } int solve() { int ret = 0; memset(bx,0,sizeof(bx)); memset(by,0,sizeof(by)); for(int i = 1;i <= cntx;i++) if(!bx[i]) { memset(vis,0,sizeof(vis)); ret += dfs(i); } return ret; } int main() { int T; scanf("%d",&T); for(int kase = 1;kase <= T;kase++) { scanf("%d%d",&n,&m); for(int i = 1;i <= n;i++) scanf("%s",buf[i] + 1); //build-graph memset(xid,0,sizeof(xid)); memset(yid,0,sizeof(yid)); memset(g,0,sizeof(g)); cntx = cnty = 0; for(int i = 1;i <= n;i++) { int flag = 0; for(int j = 1;j <= m;j++) { if(buf[i][j] == 'o') { if(flag == 0) cntx++,flag = 1; xid[i][j] = cntx; } else if(buf[i][j] == '#') flag = 0; } } for(int j = 1;j <= m;j++) { int flag = 0; for(int i = 1;i <= n;i++) { if(buf[i][j] == 'o') { if(!flag) cnty++,flag = 1; yid[i][j] = cnty; } else if(buf[i][j] == '#') flag = 0; } } for(int i = 1;i <= n;i++) { for(int j = 1;j <= m;j++) if(xid[i][j] && yid[i][j]) { g[xid[i][j]][yid[i][j]] = true; } } int ans = solve(); printf("Case :%d\n%d\n",kase,ans); } return 0; }