标签:
Time Limit: 2000MS | Memory Limit: 65536K | |
Total Submissions: 16924 | Accepted: 5284 |
Description
Input
Output
Sample Input
4 3 2 2 1 3 3
Sample Output
YES
思路:将两个相邻的grid建边,判断二分图最大匹配*2+k是否等于n*m
#include <cstdio> #include <cstring> #include <vector> using namespace std; const int MAXN=40; const int MAXV=1100; int n,m,k; int mz[MAXN][MAXN]; int dy[4]={0,1,0,-1}; int dx[4]={1,0,-1,0}; vector<int> arc[MAXV]; int vis[MAXN][MAXN]; int match[MAXV],used[MAXV]; bool dfs(int u) { for(int i=0;i<arc[u].size();i++) { int to=arc[u][i]; if(!used[to]) { used[to]=1; int w=match[to]; if(w==-1||dfs(w)) { match[to]=u; match[u]=to; return true; } } } return false; } int max_flow() { int ans=0; memset(match,-1,sizeof(match)); int limit=n*m; for(int i=1;i<=limit;i++) { if(match[i]==-1) { memset(used,0,sizeof(used)); if(dfs(i)) { ans++; } } } return ans; } int main() { while(scanf("%d%d%d",&n,&m,&k)!=EOF) { if((n*m-k)%2!=0) { printf("NO\n"); continue; } for(int i=0;i<MAXV;i++) arc[i].clear(); memset(vis,0,sizeof(vis)); memset(mz,0,sizeof(mz)); for(int i=0;i<k;i++) { int x,y; scanf("%d%d",&x,&y); mz[y][x]=-1; } //int edge=0; for(int y=1;y<=n;y++) { for(int x=1;x<=m;x++) { vis[y][x]=1; if(mz[y][x]==-1) continue; for(int i=0;i<4;i++) { int ny=y+dy[i]; int nx=x+dx[i]; if(1<=ny&&ny<=n&&1<=nx&&nx<=m&&mz[ny][nx]!=-1&&!vis[ny][nx]) { // edge++; int u=(y-1)*m+x; int v=(ny-1)*m+nx; arc[u].push_back(v); arc[v].push_back(u); } } } } //printf("%d\n",edge); int res=max_flow(); if(res*2+k==n*m) { printf("YES\n"); } else { printf("NO\n"); } } return 0; }
标签:
原文地址:http://www.cnblogs.com/program-ccc/p/5838733.html