最大流-Ford-Fulkerson增广路径算法py/cpp/Java三语言实现

article/2025/8/25 12:52:55

最大流-Ford-Fulkerson增广路径算法py/cpp/Java三语言实现

    • 一、网络流问题与相关概念
      • 1.1 网络流问题定义
      • 1.2 关键概念
    • 二、Ford-Fulkerson算法原理
      • 2.1 核心思想
      • 2.2 算法步骤
    • 三、Ford-Fulkerson算法的代码实现
      • 3.1 Python实现
      • 3.2 C++实现
      • 3.3 Java实现
    • 四、Ford-Fulkerson算法的时间复杂度与空间复杂度分析
      • 4.1 时间复杂度
      • 4.2 空间复杂度
    • 五、Ford-Fulkerson算法的应用场景
      • 5.1 资源分配
      • 5.2 通信网络
      • 5.3 交通运输
      • 5.4 任务调度与工作流管理
    • 总结

网络流问题是一类具有重要实际意义的问题,广泛应用于资源分配、通信网络、交通运输等多个领域。Ford-Fulkerson算法作为求解网络流问题的经典算法之一,基于增广路径的思想,能够有效地找到网络中的最大流。本文我将深入剖析Ford-Fulkerson算法的原理、详细介绍其实现流程,并分别使用Python、C++和Java三种语言进行代码实现,带你全面掌握这一重要算法。

一、网络流问题与相关概念

1.1 网络流问题定义

网络流问题可以抽象为一个有向图 (G=(V, E)),其中 (V) 是顶点集合,(E) 是边集合。在这个图中,存在一个源点 (s) 和一个汇点 (t),每条边 ((u, v) \in E) 都有一个非负的容量 (c(u, v)),表示这条边能够传输的最大流量。网络流问题的目标是在满足流量守恒(除源点和汇点外,每个顶点的流入流量等于流出流量)的前提下,找到从源点 (s) 到汇点 (t) 的最大流量 。

1.2 关键概念

  • 流量:对于边 ((u, v)),其流量 (f(u, v)) 表示实际通过这条边的流量,满足 (0 \leq f(u, v) \leq c(u, v))。同时,对于除源点 (s) 和汇点 (t) 之外的任意顶点 (v),流入 (v) 的总流量等于流出 (v) 的总流量,即 (\sum_{u:(u,v)\in E} f(u, v) = \sum_{w:(v,w)\in E} f(v, w))。
  • 残留网络:对于给定的网络 (G) 和流量 (f),残留网络 (G_f) 由原网络中的顶点和一些残留边组成。残留边 ((u, v)) 的容量 (c_f(u, v)) 定义为:
    • 若 ((u, v) \in E),则 (c_f(u, v) = c(u, v) - f(u, v)),表示这条边还能容纳的额外流量。
    • 若 ((v, u) \in E),则 (c_f(u, v) = f(v, u)),表示可以通过反向边撤销已经流过的流量。
    • 若 ((u, v) \notin E) 且 ((v, u) \notin E),则 (c_f(u, v) = 0)。
  • 增广路径:在残留网络 (G_f) 中,从源点 (s) 到汇点 (t) 的一条简单路径称为增广路径。沿着增广路径可以增加从源点到汇点的流量,增广路径上所有残留边的最小容量就是可以增加的流量值 。

二、Ford-Fulkerson算法原理

2.1 核心思想

Ford-Fulkerson算法基于贪心策略和增广路径的概念。算法的核心思想是从初始流量为 0 的状态开始,不断在残留网络中寻找增广路径。一旦找到增广路径,就沿着这条路径增加流量,更新残留网络。重复这个过程,直到在残留网络中不存在从源点到汇点的增广路径为止。此时,当前的流量就是网络的最大流。

2.2 算法步骤

  1. 初始化:创建原始网络 (G),设定源点 (s) 和汇点 (t),将所有边的流量 (f(u, v)) 初始化为 0,构建初始的残留网络 (G_f)。
  2. 寻找增广路径:在残留网络 (G_f) 中使用深度优先搜索(DFS)或广度优先搜索(BFS)等方法寻找从源点 (s) 到汇点 (t) 的增广路径。如果找到了增广路径,则进入步骤 3;如果找不到增广路径,说明已经达到最大流,算法结束,返回当前的流量作为最大流。
  3. 增加流量:计算增广路径上所有残留边的最小容量 (\delta),(\delta) 就是可以沿着增广路径增加的流量值。然后沿着增广路径更新原始网络中边的流量 (f(u, v)) 和残留网络中边的容量 (c_f(u, v))。对于增广路径上的边 ((u, v)):
    • 在原始网络中,若 ((u, v)) 是正向边,则 (f(u, v) = f(u, v) + \delta);若 ((u, v)) 是反向边,则 (f(u, v) = f(u, v) - \delta)。
    • 在残留网络中,更新相应边的容量 (c_f(u, v)),以反映流量的变化。
  4. 重复步骤:返回步骤 2,继续在更新后的残留网络中寻找增广路径,直到找不到增广路径为止。
    增广路径

三、Ford-Fulkerson算法的代码实现

3.1 Python实现

from collections import dequedef bfs(residual_graph, source, sink, parent):visited = [False] * len(residual_graph)queue = deque()queue.append(source)visited[source] = Truewhile queue:u = queue.popleft()for ind, val in enumerate(residual_graph[u]):if not visited[ind] and val > 0:queue.append(ind)visited[ind] = Trueparent[ind] = ureturn visited[sink]def ford_fulkerson(graph, source, sink):residual_graph = [row.copy() for row in graph]parent = [-1] * len(graph)max_flow = 0while bfs(residual_graph, source, sink, parent):path_flow = float("Inf")s = sinkwhile s != source:path_flow = min(path_flow, residual_graph[parent[s]][s])s = parent[s]max_flow += path_flowv = sinkwhile v != source:u = parent[v]residual_graph[u][v] -= path_flowresidual_graph[v][u] += path_flowv = parent[v]return max_flowgraph = [[0, 16, 13, 0, 0, 0],[0, 0, 10, 12, 0, 0],[0, 4, 0, 0, 14, 0],[0, 0, 9, 0, 0, 20],[0, 0, 0, 7, 0, 4],[0, 0, 0, 0, 0, 0]
]
source = 0
sink = 5
print(ford_fulkerson(graph, source, sink))

在上述Python代码中,bfs 函数用于在残留网络中使用广度优先搜索寻找增广路径。ford_fulkerson 函数则实现了Ford-Fulkerson算法的整体流程,包括初始化残留网络、不断寻找增广路径、增加流量并更新残留网络,直到找不到增广路径,最终返回最大流。

3.2 C++实现

#include <iostream>
#include <vector>
#include <queue>
using namespace std;bool bfs(vector<vector<int>>& residual_graph, int source, int sink, vector<int>& parent) {vector<bool> visited(residual_graph.size(), false);queue<int> q;q.push(source);visited[source] = true;while (!q.empty()) {int u = q.front();q.pop();for (int ind = 0; ind < residual_graph.size(); ind++) {if (!visited[ind] && residual_graph[u][ind] > 0) {q.push(ind);visited[ind] = true;parent[ind] = u;}}}return visited[sink];
}int ford_fulkerson(vector<vector<int>>& graph, int source, int sink) {vector<vector<int>> residual_graph = graph;vector<int> parent(graph.size(), -1);int max_flow = 0;while (bfs(residual_graph, source, sink, parent)) {int path_flow = INT_MAX;int v = sink;while (v != source) {path_flow = min(path_flow, residual_graph[parent[v]][v]);v = parent[v];}max_flow += path_flow;v = sink;while (v != source) {int u = parent[v];residual_graph[u][v] -= path_flow;residual_graph[v][u] += path_flow;v = parent[v];}}return max_flow;
}int main() {vector<vector<int>> graph = {{0, 16, 13, 0, 0, 0},{0, 0, 10, 12, 0, 0},{0, 4, 0, 0, 14, 0},{0, 0, 9, 0, 0, 20},{0, 0, 0, 7, 0, 4},{0, 0, 0, 0, 0, 0}};int source = 0;int sink = 5;cout << ford_fulkerson(graph, source, sink) << endl;return 0;
}

C++代码中,bfs 函数实现了在残留网络中使用广度优先搜索寻找增广路径的功能。ford_fulkerson 函数按照Ford-Fulkerson算法的步骤,初始化残留网络,通过不断调用 bfs 寻找增广路径,更新流量和残留网络,最终返回最大流。

3.3 Java实现

import java.util.LinkedList;
import java.util.Queue;class FordFulkerson {static boolean bfs(int[][] residualGraph, int source, int sink, int[] parent) {boolean[] visited = new boolean[residualGraph.length];Queue<Integer> queue = new LinkedList<>();queue.add(source);visited[source] = true;while (!queue.isEmpty()) {int u = queue.poll();for (int ind = 0; ind < residualGraph.length; ind++) {if (!visited[ind] && residualGraph[u][ind] > 0) {queue.add(ind);visited[ind] = true;parent[ind] = u;}}}return visited[sink];}static int fordFulkerson(int[][] graph, int source, int sink) {int[][] residualGraph = new int[graph.length][graph.length];for (int i = 0; i < graph.length; i++) {for (int j = 0; j < graph.length; j++) {residualGraph[i][j] = graph[i][j];}}int[] parent = new int[graph.length];int maxFlow = 0;while (bfs(residualGraph, source, sink, parent)) {int pathFlow = Integer.MAX_VALUE;int v = sink;while (v != source) {pathFlow = Math.min(pathFlow, residualGraph[parent[v]][v]);v = parent[v];}maxFlow += pathFlow;v = sink;while (v != source) {int u = parent[v];residualGraph[u][v] -= pathFlow;residualGraph[v][u] += pathFlow;v = parent[v];}}return maxFlow;}
}public class Main {public static void main(String[] args) {int[][] graph = {{0, 16, 13, 0, 0, 0},{0, 0, 10, 12, 0, 0},{0, 4, 0, 0, 14, 0},{0, 0, 9, 0, 0, 20},{0, 0, 0, 7, 0, 4},{0, 0, 0, 0, 0, 0}};int source = 0;int sink = 5;System.out.println(FordFulkerson.fordFulkerson(graph, source, sink));}
}

Java代码中,bfs 方法用于在残留网络中进行广度优先搜索以寻找增广路径。fordFulkerson 方法实现了Ford-Fulkerson算法的全过程,包括初始化残留网络、不断利用 bfs 寻找增广路径、更新流量和残留网络,最后返回最大流。

四、Ford-Fulkerson算法的时间复杂度与空间复杂度分析

4.1 时间复杂度

Ford-Fulkerson算法的时间复杂度取决于寻找增广路径的方法以及网络的结构。

  • 使用BFS或DFS寻找增广路径:在每次迭代中,寻找增广路径的时间复杂度为 (O(V + E)),其中 (V) 是顶点数量,(E) 是边的数量。
  • 迭代次数:算法的迭代次数取决于网络中的最大流的值 (|f^|) 和边的容量。在最坏情况下,算法可能需要进行 (|f^|) 次迭代才能找到最大流。如果边的容量都是整数,且最大流的值为 (|f^|),则算法的时间复杂度为 (O(|f^|(V + E)))。
  • 改进版本:通过使用更高效的寻找增广路径的方法(如Edmonds-Karp算法,它总是选择最短的增广路径),可以将时间复杂度改进为 (O(VE^2)),其中 (V) 是顶点数,(E) 是边数。

4.2 空间复杂度

Ford-Fulkerson算法的空间复杂度主要取决于存储网络和残留网络所需的空间:

  • 存储网络:如果使用邻接矩阵存储网络,空间复杂度为 (O(V^2));如果使用邻接表存储网络,空间复杂度为 (O(V + E))。
  • 存储残留网络:同样,使用邻接矩阵存储残留网络时空间复杂度为 (O(V^2)),使用邻接表时为 (O(V + E))。此外,还需要一些额外的空间用于存储搜索过程中的辅助数据(如访问标记、路径信息等),但这些辅助数据的空间复杂度相对较小。因此,在使用邻接表存储时,算法的空间复杂度为 (O(V + E)) 。

五、Ford-Fulkerson算法的应用场景

5.1 资源分配

在资源分配问题中,可以将资源的提供者看作源点,资源的需求者看作汇点,资源的传输路径看作边,边的容量表示路径能够传输的资源量上限。通过Ford-Fulkerson算法可以找到一种最优的资源分配方案,使得从资源提供者到需求者的资源传输总量最大,从而实现资源的合理分配 。

5.2 通信网络

在通信网络中,数据从源节点传输到目标节点,网络中的链路可以看作边,链路的带宽可以看作边的容量。使用Ford-Fulkerson算法可以计算出在给定网络拓扑和链路带宽的情况下,从源节点到目标节点的最大数据传输量,帮助网络管理员优化网络资源,提高网络的传输效率和可靠性。

5.3 交通运输

在城市交通系统中,道路可以看作边,道路的通行能力可以看作边的容量,起点和终点可以分别看作源点和汇点。通过Ford-Fulkerson算法可以分析城市交通网络的最大通行流量,为交通规划和拥堵管理提供决策依据,例如确定需要拓宽哪些道路或者调整交通流量的分配策略 。

5.4 任务调度与工作流管理

在一些任务调度系统中,任务之间可能存在依赖关系和资源限制。可以将任务看作顶点,任务之间的依赖关系和资源传输关系看作边,边的容量表示资源的限制或任务执行的先后顺序约束。Ford-Fulkerson算法可以用于确定在满足所有约束条件下,能够完成的最大任务数量或者最优的任务执行顺序,提高任务调度的效率和资源利用率。

总结

本文我从网络流基础概念出发,逐步深入讲解了Ford-Fulkerson算法的原理、多种语言实现方式,分析了其时间与空间复杂度,并探讨了丰富的实际应用场景。希望通过这些内容,你能够全面理解该算法的理论与实践价值。

That’s all, thanks for reading!
觉得有用就点个赞、收进收藏夹吧!关注我,获取更多干货~


http://www.hkcw.cn/article/DVjCqLJGJb.shtml

相关文章

摄像头模块的镜头类型

一、‌按光学功能分类‌ ‌球面镜&#xff08;Spherical Lens&#xff09;‌ ‌特点‌&#xff1a;表面为球面曲率&#xff0c;工艺简单且成本低&#xff0c;但存在球面像差和色差&#xff0c;边缘画质易模糊。 ‌应用‌&#xff1a;低端监控设备、玩具相机等对画质要求低的…

汽车EPS系统的核心:驱动芯片的精准控制原理

随着科技的飞速发展&#xff0c;电机及其驱动技术在现代工业、汽车电子、家用电器等领域扮演着越来越重要的角色。有刷马达因其结构简单、成本低廉、维护方便等优点&#xff0c;在市场上占据了一定的份额。然而&#xff0c;为了充分发挥有刷马达的性能&#xff0c;一款高效能、…

51c视觉~3D~合集3

我自己的原文哦~ https://blog.51cto.com/whaosoft/13954440 #SceneTracker 在4D时空中追踪万物&#xff01;国防科大提出首个长时场景流估计方法 本篇分享 TPAMI 2025 论文​​SceneTracker: Long-term Scene Flow Estimation Network​​&#xff0c;国防科大提出首…

【25-cv-05855】Keith律所代理Paula Alejandra Navarro 版权图

Paula Alejandra Navarro 版权图 案件号&#xff1a;25-cv-05855 立案时间&#xff1a;2025年5月27日 原告&#xff1a;Paula Alejandra Navarro 代理律所&#xff1a;Keith 原告介绍 原告是来自巴拿马的自由职业艺术家&#xff0c;擅长将精灵、中世纪服饰等经典奇幻元素…

vue自定义穿梭框(内容体+多选框)

最近需要做一个资源分配的一个功能&#xff0c;然后用到了穿梭框&#xff0c;但是需要更多的功能控制。具体业务场景如下&#xff1a;需要同时可以分配查看和下载的权限。实现效果如下&#xff1a; 组件用的是&#xff1a; Ant Design Vue 的穿梭框 操作方式&#xff1a;在左…

各国竞争的下一代液晶技术:中国铁电液晶取得重大突破突破

一、全球下一代液晶技术发展格局 &#xff08;一&#xff09;韩国&#xff1a;OLED 技术持续领先&#xff0c;布局量子点与柔性显示 韩国作为显示产业强国&#xff0c;三星、LG 等企业在 OLED 领域占据全球主导地位。三星的 AMOLED 技术广泛应用于高端智能手机&#xff0c;其柔…

张亚中提打败赖清德唯一策略 促两岸和平共识

随着国民党主席朱立伦任期即将届满,台湾孙文学校总校长张亚中已于21日宣布参选党主席。张亚中表示,国民党需要一个政策与路线的大辩论,包括两岸关系的核心问题和定位。他认为,若无此策略,国民党难以凭借资历、战力或团结击败现任台湾地区领导人赖清德重返执政。张亚中在接…

《深度关系-从建立关系到彼此信任》

陈海贤老师推荐的书&#xff0c;花了几个小时&#xff0c;感觉现在的人与人之间特别缺乏这种深度的关系&#xff0c;但是与一个人建立深度的关系并没有那么简单&#xff0c;反正至今为止&#xff0c;自己好像没有与任何一个人建立了这种深度的关系&#xff0c;那种双方高度同频…

LLaMaFactory - 支持的模型和模板 常用命令

一、 环境准备 激活LLaMaFactory环境&#xff0c;进入LLaMaFactory目录 cd LLaMA-Factoryconda activate llamafactory 下载模型 #模型下载 from modelscope import snapshot_download model_dir snapshot_download(Qwen/Qwen2.5-0.5B-Instruct) 二、启动一个 Qwen3-0.6B…

数据结构——优先级队列(PriorityQueue)

1.优先级队列 优先级队列可以看作队列的另一个版本&#xff0c;队列的返回元素是由是由插入顺序决定的&#xff0c;先进先出嘛&#xff0c;但是有时我们可能想要返回优先级较高的元素&#xff0c;比如最大值&#xff1f;这种场景下就由优先级队列登场。 优先级队列底层是由堆实…

学习如何设计大规模系统,为系统设计面试做准备!

前言 在当今快速发展的技术时代&#xff0c;系统设计能力已成为衡量一名软件工程师专业素养的重要标尺。随着云计算、大数据、人工智能等领域的兴起&#xff0c;构建高性能、可扩展且稳定的系统已成为企业成功的关键。然而&#xff0c;对于许多工程师而言&#xff0c;如何有效…

负载电容匹配:晶振电路设计中被忽视的隐形杀手

在电子电路的复杂世界里&#xff0c;晶振电路作为频率控制的核心部件&#xff0c;其稳定性和准确性对整个系统的性能起着举足轻重的作用。晶振就如同电子设备的“心脏起搏器”&#xff0c;精准地控制着电路的运行节奏。然而&#xff0c;在众多影响晶振电路性能的因素中&#xf…

Python Day36 学习

对列表、字典、元组、集合进行总结 浙大疏锦行 摘自讲义 机器学习管道Pipeline Q1. 什么是机器学习管道Pipeline&#xff1f; 摘自讲义 Q. 关于“转换器”&#xff1f; 摘自讲义 # 导入StandardScaler转换器 from sklearn.preprocessing import StandardScaler# 初始化转换…

003 flutter初始文件讲解(2)

1.书接上回 首先&#xff0c;我们先来看看昨天最后的代码及展示效果&#xff1a; import "package:flutter/material.dart";void main(){runApp(MaterialApp(home:Scaffold(appBar:AppBar(title:Text("The World")), body:Center(child:Text("Hello…

深入理解C#中的LINQ:数据查询的终极利器

在现代软件开发中&#xff0c;数据处理和查询是几乎所有应用程序的核心需求。无论是从数据库检索数据、过滤内存中的集合&#xff0c;还是解析XML文档&#xff0c;开发者都需要高效、灵活的方式来操作数据。C# 提供的 LINQ&#xff08;Language Integrated Query&#xff0c;语…

133.在 Vue3 中使用 OpenLayers 实现画多边形、任意编辑、遮罩与剪切处理功能

&#x1f3ac; 效果演示截图&#xff08;先睹为快&#xff09; ✨ 功能概览&#xff1a; ✅ 鼠标画任意形状多边形&#xff1b; ✏️ 点击“修改边界”可拖动顶点&#xff1b; &#x1f7e5; 点击“遮罩”后地图除多边形区域外变红&#xff1b; ✂️ 点击“剪切”后仅显示选…

爬虫到智能数据分析:Bright Data × Kimi 智能洞察亚马逊电商产品销售潜力

前言 电商数据分析在现代商业中具有重要的战略价值&#xff0c;通过对消费者行为、销售趋势、商品价格、库存等数据的深入分析&#xff0c;企业能够获得对市场动态的精准洞察&#xff0c;优化运营决策&#xff0c;预测市场趋势、优化广告投放、提升供应链效率&#xff0c;并通…

2025年信息素养大赛 图形化编程复赛 官方样题绘制图形答案解析

今天给大家做一下2025年全国青少年信息素养大赛 图形化编程复赛、决赛官方样题1 编程题&#xff0c;绘制图形及答案解析。 题外话&#xff1a;2024年对Scratch画笔画图考的比较多&#xff0c;例如7月20日的复赛小高组就考了4道数形结合的画图编程题&#xff0c;点击查看&#x…

ONLYOFFICE文档API:编辑器的品牌定制化

在当今数字化办公时代&#xff0c;文档编辑器已成为各类企业、组织和开发者不可或缺的工具之一。ONLYOFFICE 文档提供的功能丰富且强大的文档编辑 API&#xff0c;让开发者能够根据自己的产品需求和品牌特点&#xff0c;定制编辑器界面&#xff0c;实现品牌化展示&#xff0c;为…

【unity游戏开发——编辑器扩展】EditorApplication公共类处理编辑器生命周期事件、播放模式控制以及各种编辑器状态查询

注意&#xff1a;考虑到编辑器扩展的内容比较多&#xff0c;我将编辑器扩展的内容分开&#xff0c;并全部整合放在【unity游戏开发——编辑器扩展】专栏里&#xff0c;感兴趣的小伙伴可以前往逐一查看学习。 文章目录 前言一、监听编辑器事件1、常用编辑器事件2、示例监听播放模…