洛谷 P2057 [SHOI2007]善意的投票 网络流 最大流 最小割 FF算法

题目链接:

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;
}

 

 

 

 

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值