
3 3 4 1 2 1 3 2 1 2 2 3 3 4 1 2 1 3 2 1 3 2
Board 1 have 0 important blanks for 2 chessmen. Board 2 have 3 important blanks for 3 chessmen.
#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
int n,m;
int mat[105][105];
int vis[105];
int pre[105];
int find(int x) //2分匹配
{
int i;
for(i=1;i<=m;i++)
{
if(!vis[i] &&mat[x][i])
{
vis[i]=1;
if(pre[i]==-1 ||find(pre[i]))
{
pre[i]=x;
return 1;
}
}
}
return 0;
}
int solve()
{
int ans=0,i;
memset(pre,-1,sizeof(pre)); //每次跑2分匹配记得初始化。
for(i=1;i<=n;i++)
{
memset(vis,0,sizeof(vis));
if(find(i))
ans++;
}
return ans;
}
int main()
{
int i,k,x,y,j;
int c=0;
while(scanf("%d%d%d",&n,&m,&k)!=EOF)
{
memset(mat,0,sizeof(mat));
memset(pre,-1,sizeof(pre));
for(i=1;i<=k;i++)
{
scanf("%d%d",&x,&y);
mat[x][y]=true;
}
int sum=solve(); // 先跑个2分匹配求出最小覆盖点。
int res=0; //重要点
for(i=1;i<=n;i++)
for(j=1;j<=m;j++)
{
if(mat[i][j])
{
mat[i][j]=0; //先去掉这个点,
if(solve()!=sum) //跑完2分匹配如果不等于最小覆盖点,说明有影响,则是重要点
res++; //重要点加一。
mat[i][j]=1; //还原这个点。
}
}
printf("Board %d have %d important blanks for %d chessmen.\n",++c,res,sum);
}
return 0;
} 原文地址:http://blog.csdn.net/sky_miange/article/details/43759845