POJ 3020 Antenna Placement 二分匹配

二分图匹配问题。

一开始想不到怎么建图,参考了别人的代码之后才写出来  = - = 

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <string>
#include <cmath>
#include <cstring>
#include <queue>
#include <set>
#include <vector>
#include <stack>
#include <map>
#include <iomanip>
#define PI acos(-1.0)
#define Max 1005
#define inf 1<<28
using namespace std;

char a;
bool visit[Max];
int Map[Max][Max],match[Max];
int Map1[Max][Max];
int move[4][2]= {{0,1},{1,0},{-1,0},(0,-1)};//四个方向
int n,m;
int v1,v2;//两边点集数


//找增广轨
int dfs(int cur)
{
    int i;
    for(i=1; i<=v2; i++)
    {
        if(!visit[i]&&Map1[cur][i])
        {
            visit[i]=1;
            if(match[i]==0||dfs(match[i]))
            {
                match[i]=cur;
                return 1;
            }
        }
    }
    return 0;
}
int main()
{
    //freopen("in1.txt","w",stdout);
    int i,j,k,l;
    int T;
    while(scanf("%d",&T)!=EOF)
    {
        while(T--)
        {
            cin>>n>>m;
            int num=0;
            memset(Map,0,sizeof(Map));
            memset(Map1,0,sizeof(Map1));
            memset(match,0,sizeof(match));
            for(i=1; i<=n; i++)
            {
                for(j=1; j<=m; j++)
                {
                    cin>>a;
                    if(a=='*')
                    {
                        Map[i][j]=++num;//将每个需要覆盖的城市标号1-num;
                    }
                }
            }
//建图
            for(i=1; i<=n; i++)
            {
                for(j=1; j<=m; j++)
                {
                    if(Map[i][j])
                    {
                        for(k=0; k<4; k++)
                        {
                            int x=i+move[k][0];
                            int y=j+move[k][1];
                            if(Map[x][y])
                            {
                                Map1[Map[x][y]][Map[i][j]]=Map1[Map[i][j]][Map[x][y]]=1;//如果两个城市可以到达,则加入集合,无向图,所以两边都要加
                            }

                        }
                    }
                }
            }
            v1=v2=num;


            int sum=0;
            for(i=1; i<=v1; i++)
            {
                memset(visit,0,sizeof(visit));
                sum+=dfs(i);//找最大匹配
            }
            cout<<num-sum/2<<endl;//无向二分图:最小路径覆盖数=点的总数-最大匹配/2,这个我也不懂。。看大神是这么说的。。
        }
    }
    return 0;
}

总的来说,不能讲问题转化掉.

这题建图的想法很好,值得借鉴。

你可能感兴趣的:(POJ 3020 Antenna Placement 二分匹配)