题目链接:
https://www.luogu.com.cn/problem/P2057
参考思路博客:
https://www.luogu.com.cn/blog/motherfucker-mengyu/solution-p2057
算法:1:FF算法
思路:
1:最小割等于最大流
2:这道题目是求最小割,两种意见可以看作源点S和T,我们需要做的是割最少的边使得S和T成为两个不同的集合,解释:割掉的边相当于1次冲突(因为若某边没被割掉,则显然这条边相连的两个点分别通向了S和T,所以算是一次冲突),当S和T还连通时则必然存在一条路径,这样肯定会有冲突,所以需要使得S和T孤立。所以就是求这道题目的最小割,而最小割就等于最大流,所以这道题最大流跑一跑就好了
3:要建一个超级源点s以及一个超级汇点t,将其他的点分别连上这两个点
注意:
1:朋友之间是双向边,权值均为1
#include <bits/stdc++.h>
using namespace std;
const int maxn=3e2+2,maxm=3e2*(3e2+3)+1;
int n,m,a,b,s,t,tot=1,head[maxn],vis[maxn];
struct edge
{
int to,next,w;
}e[maxm];
inline void addedge(int x,int y,int w)
{
e[++tot].to=y;
e[tot].w=w;
e[tot].next=head[x];
head[x]=tot;
}
inline int dfs(int x,int flow)
{
if(x==t)return flow;
vis[x]=1;
for(int i=head[x];i;i=e[i].next)
{
int y=e[i].to,w=e[i].w;
if(w&&!vis[y])
{
int t=dfs(y,min(flow,w));
if(t>0)
{
e[i].w-=t;
e[i^1].w+=t;
return t;
}
}
}
return 0;
}
int main()
{
ios::sync_with_stdio(0);
scanf("%d %d",&n,&m);
s=0,t=n+1;
for(int i=1;i<=n;i++)
{
scanf("%d",&a);
if(a)addedge(s,i,1),addedge(i,s,0);
else addedge(i,t,1),addedge(t,i,0);
}
for(int i=1;i<=m;i++)scanf("%d %d",&a,&b),addedge(a,b,1),addedge(b,a,1);
int res=0,ans=0;
while(memset(vis,0,sizeof(vis))&&(res=dfs(s,1e9))>0)ans+=res;
printf("%d\n",ans);
return 0;
}