二分图匹配问题。
一开始想不到怎么建图,参考了别人的代码之后才写出来 = - =
#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;
}
总的来说,不能讲问题转化掉.
这题建图的想法很好,值得借鉴。