AcWing算法进阶课-1.9.1Dinic/ISAP求最小割

算法进阶课整理

CSDN个人主页:更好的阅读体验

Start

原题链接
题目描述

给定一个包含 n n n 个点 m m m 条边的有向图,并给定每条边的容量,边的容量非负。

图中可能存在重边和自环。求从点 S S S 到点 T T T 的最小割。

输入格式

第一行包含四个整数 n , m , S , T n,m,S,T n,m,S,T

接下来 m m m 行,每行三个整数 u , v , c u,v,c u,v,c,表示从点 u u u 到点 v v v 存在一条有向边,容量为 c c c

点的编号从 1 1 1 n n n

输出格式

输出点 S S S 到点 T T T 的最小割。

如果从点 S S S 无法到达点 T T T 则输出 0 0 0

数据范围

2 ≤ n ≤ 10000 2 \le n \le 10000 2n10000,
1 ≤ m ≤ 100000 1 \le m \le 100000 1m100000,
0 ≤ c ≤ 10000 0 \le c \le 10000 0c10000,
S ≠ T S \neq T S=T


思路

最大流最小割定理

对于任意一个流网络 G = ( V , E ) G=(V,E) G=(V,E) 都满足:

  1. 可行流 f f f 是最大流。

⇔ \Leftrightarrow

  1. 可行流 f f f 的残量网络中不存在增广路。

⇔ \Leftrightarrow

  1. ∃ [ S , T ] , ∣ f ∣ = c ( S , T ) \exist [\text S,\text T],|f|=c(\text S,\text T) [S,T],f=c(S,T)

证明

  • 1 ⇒ 2 1\Rightarrow 2 12:反证法。假设 f f f 是最大流,且 G f G_f Gf 存在增广路径。这说明 G f G_f Gf 存在至少一个流量大于 0 0 0 的可行流 f ′ f' f。这样能构造出原网络中另一个可行流 f + f ′ f+f' f+f,且 ∣ f + f ′ ∣ > ∣ f ∣ |f+f'|>|f| f+f>f,说明 f f f 不是最大流,与假设矛盾。
  • 2 ⇒ 3 2\Rightarrow 3 23:构造一个割,使得它在残量网络中不存在增广路径,且 ∣ f ∣ = c ( S , T ) |f|=c(\text S,\text T) f=c(S,T)。定义集合 S \text S S 为在 G f G_f Gf 中从 S S S 出发,沿容量大于 0 0 0 的边走能走到的所有的点。由于残量网络中不存在增广路径,所以集合 S \text S S 中不可能包含 T T T。在定义集合 T = V − S \text T = \text V - \text S T=VS,即为构造的合法割。对于正向边 ( u , v ) [ u ∈ S , v ∈ T ] (u,v)[u\in \text S,v\in \text T] (u,v)[uS,vT],由于 u , v u,v u,v 不相通,所以 f ′ ( u , v ) = 0 f'(u, v)=0 f(u,v)=0,所以 f ( u , v ) = c ( u , v ) f(u,v)=c(u,v) f(u,v)=c(u,v)。对于反向边 ( u , v ) [ u ∈ T , v ∈ S ] (u,v)[u\in \text T,v\in \text S] (u,v)[uT,vS] 同理, c ′ ( v , u ) = 0 c'(v,u)=0 c(v,u)=0,即 f ( u , v ) = 0 f(u,v)=0 f(u,v)=0。发现原网络中 ∀ ( u , v ) ( u ∈ S , v ∈ T ) \forall (u,v)(u\in\text S,v\in\text T) (u,v)(uS,vT),都有 f ( u , v ) = c ( u , v ) , f ( v , u ) = 0 f(u,v)=c(u,v),f(v,u)=0 f(u,v)=c(u,v),f(v,u)=0。因此 ∣ f ∣ = f ( S , T ) = ∑ u ∈ S ∑ v ∈ T f ( u , v ) − ∑ u ∈ S ∑ v ∈ T f ( v , u ) = ∑ u ∈ S ∑ v ∈ T f ( u , v ) = ∑ u ∈ S ∑ v ∈ T c ( u , v ) = c ( S , T ) |f|=f(\text S,\text T)=\sum_{u\in\text S}\sum_{v\in\text T}f(u,v)-\sum_{u\in\text S}\sum_{v\in\text T}f(v,u)=\sum_{u\in\text S}\sum_{v\in\text T}f(u,v)=\sum_{u\in\text S}\sum_{v\in\text T}c(u,v)=c(\text S,\text T) f=f(S,T)=uSvTf(u,v)uSvTf(v,u)=uSvTf(u,v)=uSvTc(u,v)=c(S,T)
  • 3 ⇒ 1 3\Rightarrow 1 31:由 本文 最小割的推论中已知 ∣ f ∣ ≤ c ( S , T ) |f|≤c(\text S,\text T) fc(S,T),且最大流是可行流的一种,所以最大流 ≤ c ( S , T ) \leq c(\text S,\text T) c(S,T)。易知 ∣ f ∣ ≤ |f|\leq f 最大流,而 ∣ f ∣ = c ( S , T ) ≥ |f|=c(\text S,\text T)\geq f=c(S,T) 最大流。即 ∣ f ∣ ≥ |f|\geq f 最大流。综上得出 ∣ f ∣ = |f|= f= 最大流。已证最大流 ≤ \leq 最小割,而最小割又是割的容量的最小值,得出最小割 ≤ c ( S , T ) = ∣ f ∣ ≤ \leq c(\text S,\text T)=|f|\leq c(S,T)=f 最大流,即最小割 ≤ \leq 最大流,最终得出最大流 = 最小割。

因此要求最小割,只需求原图的最大流即可。


Dinic 算法其实是 EK 算法的一个暴力的优化,EK 算法每次只能搜索一条增广路径,而 Dinic 算法每次都用 DFS 的形式尽可能多的搜索增广路径。

而图中可能存在环,为了保证 DFS 的过程中不会造成死循环,这里可以使用分层图,这样每次都是一层一层往下搜索,就不会出现死循环。

  1. BFS 建立分层图
  2. DFS 找出所有能增广的路径
  3. 累加最大流量

注意: Dinic 算法对于优化非常敏感,如果优化的不好就可能直接 TLE

算法时间复杂度 O ( n 2 m ) O(n^2m) O(n2m)
AC Code

C + + \text{C}++ C++

#include <iostream>
#include <cstring>using namespace std;const int N = 10010, M = 200010;
const int INF = 1e9;int n, m, S, T;
int h[N], e[M], ne[M], f[M], idx;
int d[N], q[N], cur[N];
// d[] 存分层图中每个点的深度
// q[] 手写队列
// cur[] 当前弧优化inline void add(int a, int b, int c)
{e[idx] = b, f[idx] = c, ne[idx] = h[a], h[a] = idx ++ ; // 正向边e[idx] = a, f[idx] = 0, ne[idx] = h[b], h[b] = idx ++ ; // 反向边,用于反悔
}bool bfs()
{memset(d, -1, sizeof d); // 记得初始化int hh = 0, tt = 0;q[0] = S, d[S] = 0, cur[S] = h[S];// 将源点加入队列,源点深度为0,初始化源点当前弧为表头while (hh <= tt){int t = q[hh ++ ];for (int i = h[t]; ~i; i = ne[i]){int j = e[i];if (d[j] == -1 && f[i]) // 存在增广路才能从这里分层{d[j] = d[t] + 1; // 更新深度cur[j] = h[j]; // 所有点当前弧最开始都是表头if (j == T) return true; // 如果找到汇点了就说明有增广路q[ ++ tt] = j;}}}return false; // 没有增广路
}int find(int u, int lim) // DFS,u是当前点,lim是u之前的路径能流过的最大值
{if (u == T) return lim; // 如果已经流到汇点了就返回这个最大值int flow = 0; // 当前往下流的流量for (int i = cur[u]; ~i && flow < lim; cur[u] = i, i = ne[i])// 注意应从当前弧开始遍历,并且每次要更新。没有剩余流量也应该退出{int j = e[i];if (d[j] == d[u] + 1 && f[i])// 按分层图遍历,防止死循环{int t = find(j, min(f[i], lim - flow));// 找到后面能流的最大值if (!t) d[j] = -1;// 如果没有流量那么说明后面没有增广路了,这个点不会用到,将深度设为-1f[i] -= t, f[i ^ 1] += t, flow += t;// 当前边减流量,反向边加流量,实际流量加流量}}return flow;
}int dinic()
{int r = 0, flow = 0;while (bfs()) while ((flow = find(S, INF))) r += flow;// 只要存在增广路就一直DFS,直到DFS出的流量为0return r;
}int main()
{int a, b, c;memset(h, -1, sizeof h);scanf("%d%d%d%d", &n, &m, &S, &T);while (m -- ){scanf("%d%d%d", &a, &b, &c);add(a, b, c);}printf("%d\n", dinic());return 0;
}

228aa7bed3e021faf24cf8560d3e47bb.gif

最后,如果觉得对您有帮助的话,点个赞再走吧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/221176.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【08】GeoScene产品发布海图服务——以s57数据标准为例

在GeoScene产品中发布海图服务——以s57数据标准为例&#xff0c;发布的服务方便不同的客户终端调用&#xff0c;例如&#xff1a;web端通过JS api进行调用&#xff0c;移动端通过GeoScene Runtime SDK进行调用。1、海图服务部署 GeoScene_Maritime_for_Server海图模块安装完之…

JDK各个版本特性讲解-JDK14特性

JDK各个版本特性讲解-JDK14特性 一、Java14概述二、语法层面的变化1. instanceof2. switch表达式3. 文本块的改进4. Records记录类型 二、关于GC1.G1的NUMA内存分配优化2. 弃用SerialCMS,ParNewSerial Old3.删除CMS4.ZGC on macOS and Windows 三、其他变化1.友好的空指针异常提…

融了超24亿元舍不得花,放银行吃利息,这家存储创企厉害了

引言&#xff1a;AI与大模型风起云涌&#xff0c;为什么催生了这匹存储“黑马”&#xff1f; 【阿明观察 &#xff5c; 科技热点关注】 这家总部设在美国的存储初创公司&#xff0c;真的赶上AI与大模型时代的风口了。Vast Data公司最新再次获得E轮融资1.18亿美元&#xff0c;…

Navicat16的下载与安装

Navicat16的下载与安装 1、官网下载地址&#xff1a;https://www.navicat.com.cn/download/navicat-premium 当然有的朋友在官网下载比较慢&#xff0c;我也为大家准备好了百度网盘链接 链接&#xff1a;https://pan.baidu.com/s/1dUcTSHr3761Oayh0-WfolA?pwdwfpl 提取码&am…

HTML有哪些列表以及具体的使用!!!

文章目录 一、HTML列表二、列表的应用1、无序列表2、有序列表3、自定义列表 三、总结 一、HTML列表 html的列表有三种&#xff0c;一种是无序列表&#xff0c;一种是有序列表&#xff0c;还有一种为自定义列表。 二、列表的应用 1、无序列表 <ul> <li>无序列表…

实验4.2 默认路由和浮动静态路由的配置

实验4.2 默认路由和浮动静态路由的配置 一、任务描述二、任务分析三、具体要求四、实验拓扑五、任务实施1.路由器的基本配置。2.配置默认路由&#xff0c;实现全网互通。3.配置浮动静态路由&#xff0c;实现链路备份。 六、任务验收七、任务小结八、知识链接1&#xff0e;默认路…

css实现边框彩虹跑马灯效果

效果展示 代码实战 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthdevice-…

如何在公网环境下使用Potplayer访问本地群晖webdav中的影视资源

文章目录 本教程解决的问题是&#xff1a;按照本教程方法操作后&#xff0c;达到的效果是&#xff1a;1 使用环境要求&#xff1a;2 配置webdav3 测试局域网使用potplayer访问webdav3 内网穿透&#xff0c;映射至公网4 使用固定地址在potplayer访问webdav ​ 国内流媒体平台的内…

API资源对象StorageClass;Ceph存储;搭建Ceph集群;k8s使用ceph

API资源对象StorageClass;Ceph存储;搭建Ceph集群;k8s使用ceph API资源对象StorageClass SC的主要作用在于&#xff0c;自动创建PV&#xff0c;从而实现PVC按需自动绑定PV。 下面我们通过创建一个基于NFS的SC来演示SC的作用。 要想使用NFS的SC&#xff0c;还需要安装一个NFS…

23年12月AI烟火识别系统应用案例-北京梅兰芳故居防火系统

AI烟火识别智能视频分析系统在文化遗产保护领域的应用&#xff0c;尤其是在梅兰芳故居防火系统的部署&#xff0c;是现代科技与传统文化保护结合的典范。这篇文章将详细介绍富维烟火识别系统的设计、实施及其在23年12月在北京梅兰芳故居中的应用。 背景介绍 ● 梅兰芳故居的重要…

蓝牙物联网开发与应用:五大核心应用场景!

蓝牙技术在物联网中的五大核心应用场景 1、智能家居 通过蓝牙连接智能家居设备&#xff0c;如智能灯泡、智能插座、智能恒温器等&#xff0c;可以实现远程控制、语音控制等功能&#xff0c;提高家居的智能化程度和便利性。 2、智能穿戴设备 蓝牙技术可以连接智能手表、智能手…

使用MybatisPlus置空某些指定字段

当前的MybatisPlus默认会对空实体内的字段不置空&#xff0c;所以才引出了此种方法&#xff0c;很方便简单&#xff1a; 使用 Wrappers.lambdaUpdate方法就可以解决&#xff0c;方法的源码如下&#xff1a;条件为entity内的值&#xff0c;使用lambdaUpdate去set空的值 举个例子…

故障排查:shell脚本输出乱码

博客主页&#xff1a;https://tomcat.blog.csdn.net 博主昵称&#xff1a;农民工老王 主要领域&#xff1a;Java、Linux、K8S 期待大家的关注&#x1f496;点赞&#x1f44d;收藏⭐留言&#x1f4ac; 目录 故障详情故障原因解决方法iconv命令介绍 故障详情 最近的工作中遇到一…

电机驱动开发

最近在搞电机驱动程序&#xff0c;感觉很简单&#xff0c;实际操作却发现里面还有很多猫腻&#xff08;细节&#xff09;。 电机在嵌入式设备中非常常见&#xff0c;例如云台的转动&#xff0c;都是靠电机来驱动的。 电机常见分步进电机、直流电机&#xff0c;相对来说步进电机…

GaussDB数据库表创建行访问控制策略

目录 一、前言 二、GaussDB中的行访问控制 1、CREATE ROW LEVEL SECURITY POLICY语法 2、ALTER ROW LEVEL SECURITY POLICY语法 3、ROW LEVEL SECURITY策略与适配SQL语法关系 三、GaussDB中的行访问控制策略示例 1、实现GaussDB行访问控制的一般步骤 2、行访问控制策略…

【Proteus仿真】【Arduino单片机】光照强度检测系统

文章目录 一、功能简介二、软件设计三、实验现象联系作者 一、功能简介 本项目使用Proteus8仿真Arduino单片机控制器&#xff0c;使共阳数码管&#xff0c;ADC模块、光敏传感器等。 主要功能&#xff1a; 系统运行后&#xff0c;数码管显示光传感器采集光照强度值&#xff0c;…

【深度学习目标检测】九、基于yolov5的路标识别(python,目标检测)

YOLOv5是目标检测领域一种非常优秀的模型&#xff0c;其具有以下几个优势&#xff1a; 1. 高精度&#xff1a;YOLOv5相比于其前身YOLOv4&#xff0c;在目标检测精度上有了显著的提升。YOLOv5使用了一系列的改进&#xff0c;如更深的网络结构、更多的特征层和更高分辨率的输入图…

MyBatis的ORM映射

目录 什么是ORM 一&#xff0c;列的别名 二&#xff0c;结果映射 三&#xff0c;总结 什么是ORM ORM&#xff1a;对象关系映射&#xff08;Object Relational Mapping&#xff0c;简称ORM&#xff09;模式是一种为了解决面向对象与关系数据库存在的互不匹配的现象的技术。简…

51单片机应用从零开始(十一)·数组函数、指针函数

51单片机应用从零开始&#xff08;九&#xff09;数组-CSDN博客 51单片机应用从零开始&#xff08;十&#xff09;指针-CSDN博客 目录 1. 用数组作函数参数控制流水花样 2. 用指针作函数参数控制 P0 口 8 位 LED 流水点亮 1. 用数组作函数参数控制流水花样 要在51单片机中…

使用matlab制作声音采样率转换、播放以及显示的界面

利用matlab做一个声音采样率转换、播放以及显示的界面 大抵流程&#xff1a; 图形界面创建&#xff1a;使用figure函数创建名为“声音采样率转换”的图形界面&#xff0c;并设置了其位置和大小。 按钮和文本框&#xff1a;使用uicontrol函数创建了选择音频文件的按钮、显示当前…