1281棋盘游戏(二分匹配+删边)


不能放车的地方不影响车的互相攻击。
因为不影响,所以可以以整行,整列为单位建图
因为要删边,矩阵方便点
求最大匹配数,但是这题还有一个要求是:
求有多少个关键点,该关键点是指:若没有这一点则“车”放的个数将减少(即:匹配数量减少)。
那就模拟下,原来有边的点除去,检测最大匹配数是否和原来一样
如果一样,则不是关键点,如果是,则是关键点,累加

#include<iostream>
#include<string.h>
using namespace std;
int map[110][110],vis[500],pp[500];
int n,m,k;
int find_match(int u)
{
    for(int i=1;i<=m;i++)
    {
         if(map[u][i] && vis[i]==0)
         {//cout<<"map["<<u<<"]["<<i<<"]= "<<map[u][i]<<"   "<<n+i<<endl;
             vis[i] = 1;
             if(pp[n+i]==0 || find_match(pp[n+i]))
             {
                 pp[u] = i;  pp[n+i] = u;  //cout<<u<<" ---- "<<i<<endl;
                 return 1;
             }
         }
    }
    return 0;
}
int find_path()
{
    int match = 0;
    memset(pp,0,sizeof(pp));
    for(int i=1;i<=n;i++)
    {
         memset(vis,0,sizeof(vis));
         match += find_match(i); 
    }
    return match;
}
int main()
{
    int t=0;
    int  a, b;
    while(scanf("%d%d%d",&n,&m,&k)!=EOF)
    {
         memset(map,0,sizeof(map));
         for(int i=0;i<k;i++)
         {
              scanf("%d%d",&a,&b);
              map[a][b] = 1;
         }
         int match = find_path(); // cout<<"match=  "<<match<<endl;
         int sum = 0 , temp;
         for(int i=1;i<=n;i++)
         {
              for(int j=1;j<=m;j++)
              {
                  if(map[i][j])
                  {//cout<<"i= "<<i<<" j= "<<j<<endl;
                       map[i][j] = 0;
                       temp = find_path();  //cout<<"temp= "<<temp<<endl;
                       if(temp!=match) {sum++;}
                       map[i][j] = 1; 
                  }
              }
         }
         printf("Board %d have %d important blanks for %d chessmen.\n",++t,sum,match);
    }
    return 0;
} 


你可能感兴趣的:(游戏,Path)