当前位置: 首页 > news >正文

网络流 最小割 Dinic算法

image

标准模板

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=210,M=5e3+10;
int n,m,s,t,d[N],cur[N],vis[N];
int h[N],e[M<<1],ne[M<<1],id=1;//从2,3开始配对
LL c[M<<1];
void add(int u,int v,LL w){++id;e[id]=v;ne[id]=h[u];c[id]=w;h[u]=id;
}LL dfs(int u,LL mf){//多路增广if(u==t)return mf;LL sum=0;for(int i=cur[u];i;i=ne[i]){cur[u]=i;//当前弧优化int v=e[i];if(d[v]==d[u]+1&&c[i]){LL f=dfs(v,min(mf,c[i]));sum+=f;//累加u的流出流量c[i]-=f;c[i^1]+=f;//更新残留网mf-=f;//减少u的剩余流量if(mf==0)//余量优化break;}}if(sum==0)//残枝优化d[u]=0;return sum;
}bool bfs(){//对点分层,找增广路memset(d,0,sizeof(d));queue<int> q;q.push(s);d[s]=1;while(q.size()){int u=q.front();q.pop();for(int i=h[u];i;i=ne[i]){int v=e[i];if(d[v]==0&&c[i]){d[v]=d[u]+1;q.push(v);if(v==t)return true;}}}return false;
}LL dinic(){//累加可行流LL flow=0;while(bfs()){memcpy(cur,h,sizeof(h));flow+=dfs(s,1e9);}return flow;
}void mincut(int u){vis[u]=1;for(int i=h[u];i;i=ne[i]){int v=e[i];if(!vis[v]&&c[i])mincut(v);}
}void reset_edges(){//每条边是成对存储,正边id为偶数,反边id为奇数for(int i=2;i<=id;i++){if(i&1){c[i]=0;}else{if(c[i])c[i]=1e9;elsec[i]=1;}}
}
int main(){cin.tie(nullptr)->sync_with_stdio(false);cin>>n>>m>>s>>t;for(int i=1;i<=m;i++){LL u,v,w;cin>>u>>v>>w;add(u,v,w);add(v,u,0);}//最小割cout<<dinic()<<endl;//最小割划分mincut(s);//访问过的点为S集合for(int i=1;i<=n;i++){if(vis[i])cout<<i<<' ';}cout<<endl;//没访问过的点为T集合for(int i=1;i<=n;i++){if(!vis[i])cout<<i<<' ';}cout<<endl;//求最小割的最少边数//重建边时应当把第一遍dinic中剩余容量为0的正向边的边权设为1,//其他正向边设为无穷大,反向边都设为零,//因为只有流满的边才是最小割中的边。reset_edges();cout<<dinic()<<endl;return 0;
}

模板题:洛谷p1344

code

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=210,M=5e3+10;
int n,m,s,t,d[N],cur[N],vis[N];
int h[N],e[M<<1],ne[M<<1],id=1;//从2,3开始配对
LL c[M<<1];
void add(int u,int v,LL w){++id;e[id]=v;ne[id]=h[u];c[id]=w;h[u]=id;
}LL dfs(int u,LL mf){//多路增广if(u==t)return mf;LL sum=0;for(int i=cur[u];i;i=ne[i]){cur[u]=i;//当前弧优化int v=e[i];if(d[v]==d[u]+1&&c[i]){LL f=dfs(v,min(mf,c[i]));sum+=f;//累加u的流出流量c[i]-=f;c[i^1]+=f;//更新残留网mf-=f;//减少u的剩余流量if(mf==0)//余量优化break;}}if(sum==0)//残枝优化d[u]=0;return sum;
}bool bfs(){//对点分层,找增广路memset(d,0,sizeof(d));queue<int> q;q.push(s);d[s]=1;while(q.size()){int u=q.front();q.pop();for(int i=h[u];i;i=ne[i]){int v=e[i];if(d[v]==0&&c[i]){d[v]=d[u]+1;q.push(v);if(v==t)return true;}}}return false;
}LL dinic(){//累加可行流LL flow=0;while(bfs()){memcpy(cur,h,sizeof(h));flow+=dfs(s,1e9);}return flow;
}void mincut(int u){vis[u]=1;for(int i=h[u];i;i=ne[i]){int v=e[i];if(!vis[v]&&c[i])mincut(v);}
}void reset_edges(){//每条边是成对存储,正边id为偶数,反边id为奇数for(int i=2;i<=id;i++){if(i&1){c[i]=0;}else{if(c[i])c[i]=1e9;elsec[i]=1;}}
}
int main(){cin.tie(nullptr)->sync_with_stdio(false);// cin>>n>>m>>s>>t;cin>>n>>m;s=1;t=n;for(int i=1;i<=m;i++){LL u,v,w;cin>>u>>v>>w;add(u,v,w);add(v,u,0);}//最小割// cout<<dinic()<<endl;cout<<dinic()<<' ';//最小割划分// mincut(s);//访问过的点为S集合// for(int i=1;i<=n;i++){//     if(vis[i])//         cout<<i<<' ';// }// cout<<endl;//没访问过的点为T集合// for(int i=1;i<=n;i++){//     if(!vis[i])//         cout<<i<<' ';// }// cout<<endl;//求最小割的最少边数//重建边时应当把第一遍dinic中剩余容量为0的正向边的边权设为1,//其他正向边设为无穷大,反向边都设为零,//因为只有流满的边才是最小割中的边。reset_edges();cout<<dinic()<<endl;return 0;
}
http://www.jsqmd.com/news/7395/

相关文章:

  • 15.VLANIF(2025年9月30日) - 教程
  • 国庆集训DAY2
  • 详细介绍:电子电气架构 --- 中国汽车座舱产品与技术发展趋势展望
  • 新手Markdown学习
  • 马云归来,“新零售”不死 - 指南
  • SQL正则表达式总结 - 实践
  • Pdfminer-Vulnerability-Research
  • 10.2笔记
  • Shell / Bash 学习
  • 【Linux 架构探幽:从入门到内核・系统编程开篇】基础指令与权限精讲,筑牢框架制作根基
  • chisel,spatial和spinalhdl的比较
  • 国庆集训Day1
  • ChIPBase network菜单 生成tf的excel ,用于构建 TF → mRNA(即 CDKN3)调控关系的详细过程和教程 - 实践
  • 实用指南:机器学习:线性回归
  • 定时任务详解
  • Linux系统中配置SSH安全和Docker安装
  • Markdown语法入门三:链接,图片,分隔线与引用
  • 华为wlan无线配置 - 教程
  • 开源 C# 飞快研发(十三)进程--管道通讯
  • Spring Boot 内置日志框架 Logback - 以及 lombok 介绍 - 教程
  • PINN训练新思路:把初始条件和边界约束嵌入网络架构,解决多目标优化难题
  • 图的匹配
  • Tarjan 算法
  • 临项交换
  • CF VP 记录
  • 华为设备MSTP - 指南
  • LabVIEW与PLC 汽车驻车制动自动调整 - 实践
  • 04. 布局管理
  • 关于安装博客园皮肤中有关于配置音乐播放器的补充(awescnb)
  • AGC VP 记录 2