MinCostMaxFlow-Graph Algorithm

lab要求如下:

1.代码实现思路

  1. 图的构建

    • 使用邻接表 adjacencyList 来存储图的结构,每个节点对应一个列表,列表中存储从该节点出发的所有边。

    • 通过 addEdge 方法添加有向边及其反向边,同时设置正向边和反向边的相互引用。

  2. 最小费用最大流算法

    • 算法主体在 minCostMaxFlow 方法中,使用循环不断寻找增广路径,直到无法找到从源点到汇点的路径为止。

    • 在每次循环中,使用 Dijkstra 算法来寻找从源点到汇点的最小费用路径,通过维护距离数组 dist、前驱边数组 parent 和优先队列 pq 来实现。

    • 找到最小费用路径后,确定增广路径上的最小剩余容量 flow,然后沿着增广路径更新流量和费用。wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

	static final int INF = Integer.MAX_VALUE; // 无穷大
    // 边的定义
    static class Edge {
        int start, end, capacity, cost, flow; // 起点、终点、容量、费用、流量
        Edge residual; // 反向边

        public Edge(int start, int end, int capacity, int cost) {
            this.start = start;
            this.end = end;
            this.capacity = capacity;
            this.cost = cost;
            this.flow = 0;
        }

        // 剩余容量
        int remainingCapacity() {
            return capacity - flow;
        }

        // 增加流量
        void augmentFlow(int value) {
            this.flow += value;
            this.residual.flow -= value;
        }
    }

    static int nodeCount, edgeCount, source, sink; // 节点数、边数、源点、汇点
    static List<Edge>[] adjacencyList; // 邻接表

    // 添加有向边和反向边
    static void addEdge(int from, int to, int capacity, int cost) {
        if (capacity < 0) {
            throw new IllegalArgumentException("Capacity cannot be negative.");
        }
        Edge forwardEdge = new Edge(from, to, capacity, cost);
        Edge backwardEdge = new Edge(to, from, 0, -cost);
        forwardEdge.residual = backwardEdge;
        backwardEdge.residual = forwardEdge;
        adjacencyList[from].add(forwardEdge); // 正向边
        adjacencyList[to].add(backwardEdge); // 反向边
    }

关键算法:

整体思路:通过不断寻找最小费用增广路径并更新流量和费用,直到不存在从源点到汇点的增广路径为止。找到一条从源节点到汇点的路径。计算路径上的最小容量,并更新每个边的流量。沿路径反向更新每个边的残余容量。如果找不到一条路径,则算法结束。否则,重复循环,直到找不到更多路径,计算最大流和最小成本。

 // 最小费用最大流算法
    static int[] minCostMaxFlow() {
        int totalFlow = 0, totalCost = 0;

        while (true) {//Dijkstra算法
            int[] dist = new int[nodeCount + 1];
            Edge[] parent = new Edge[nodeCount + 1];
            Arrays.fill(dist, INF);
            dist[source] = 0;
            boolean[] visited = new boolean[nodeCount + 1]; // 访问标记
            PriorityQueue<Integer> pq = new PriorityQueue<>(Comparator.comparingInt(a -> dist[a]));
            pq.add(source); // 只添加源点

            while (!pq.isEmpty()) {
                int u = pq.poll();
                // 如果该节点已被访问,跳过
                if (visited[u]) continue;
                visited[u] = true; // 标记为已访问
                for (Edge edge : adjacencyList[u]) {
                    if (edge.remainingCapacity() > 0 && dist[edge.end] > dist[u] + edge.cost) {
                        dist[edge.end] = dist[u] + edge.cost;
                        parent[edge.end] = edge;
                        // 只有在未访问的情况下才添加到优先队列
                        if (!visited[edge.end]) {
                            pq.add(edge.end);
                        }
                    }
                }
            }
            // 如果无法到达汇点,退出
            if (dist[sink] == INF) break;

            // 找到增广流量
            int flow = INF;
            for (int v = sink; v != source; v = parent[v].start) {
                flow = Math.min(flow, parent[v].remainingCapacity());
            }
            // 增广路径上的流量与费用更新
            for (int v = sink; v != source; v = parent[v].start) {
                parent[v].augmentFlow(flow);
                totalCost += flow * parent[v].cost;
            }
            totalFlow += flow;
        }
        return new int[]{totalFlow, totalCost}; // 直接返回最大流和最小费用
    }

运行结果示例:

2.复杂度分析

  • 时间复杂度:每次执行 Dijkstra 算法的时间复杂度是 O((E + V) log V),而最多执行 O(F) 次。综合时间复杂度为:O(F⋅(E+V)log⁡V)。其中 F 是最大流量,E 是边的数量,V 是节点数量。

  • 空间复杂度:空间复杂度主要由邻接表,dist[] 数组,parent[] 数组,visited[] 数组,PriorityQueue,边的存储构成。因此,总体空间复杂度为O(V+E)

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

简单工厂模式和策略模式的异同

文章目录 简单工厂模式和策略模式的异同相同点&#xff1a;不同点&#xff1a;目的&#xff1a;结构&#xff1a; C 代码示例简单工厂模式示例&#xff08;以创建图形对象为例&#xff09;策略模式示例&#xff08;以计算价格折扣策略为例&#xff09;UML区别 简单工厂模式和策…

欢迎 PaliGemma 2 – 来自 Google 的新视觉语言模型

我们很高兴迎来 Google 全新的视觉语言模型 PaliGemma 2&#xff0c;这是 PaliGemma 的一个新版本。与其前代产品一样&#xff0c;PaliGemma 2 使用强大的SigLIP进行视觉处理&#xff0c;但在文本解码部分升级到了最新的 Gemma 2。 https://hf.co/collections/google/siglip-65…

Django基础 - 01入门简介

一、 基本概念 1.1 Django说明 Django发布于2005年&#xff0c; 网络框架&#xff0c; 用Python编写的开源的Web应用框架。采用了MVC框架模式&#xff0c;也称为MTV模式。官网&#xff1a; https://www.djangoproject.com1.2 MVC框架 Model&#xff1a; 封装和数据库相关…

华为OD --- 敏感字段加密

华为OD --- 敏感字段加密 题目独立实现思路源码实现 参考实现 题目 独立实现 思路 通过便利字符串把所有“关键字”找出来,然后将第N个关键字替换成******,最后再通过 “_” 拼接起来即可 源码实现 const rl require("readline").createInterface({ input: proce…

WebRTC服务质量(05)- 重传机制(02) NACK判断丢包

WebRTC服务质量&#xff08;01&#xff09;- Qos概述 WebRTC服务质量&#xff08;02&#xff09;- RTP协议 WebRTC服务质量&#xff08;03&#xff09;- RTCP协议 WebRTC服务质量&#xff08;04&#xff09;- 重传机制&#xff08;01) RTX NACK概述 WebRTC服务质量&#xff08;…

着色器 (三)

今天&#xff0c;是我们介绍opengl着色器最后一章&#xff0c;着色器(Shader)是运行在GPU上的小程序。这些小程序为图形渲染管线的某个特定部分而运行。从基本意义上来说&#xff0c;着色器只是一种把输入转化为输出的程序。着色器也是一种非常独立的程序&#xff0c;因为它们之…

【Linux网络】网络基础:IP协议

&#x1f4dd;个人主页&#x1f339;&#xff1a;Eternity._ ⏩收录专栏⏪&#xff1a;Linux “ 登神长阶 ” &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; ❀ IP协议 IP协议基本概念协议头格式分片与组装网段划分子网掩码特殊的IP地址 IP地址的数量限制…

neo4j 图表数据导入到 TuGraph

neo4j 图表数据导入到 TuGraph 代码文件说明后文 前言:近期在引入阿里的 TuGraph 图数据库&#xff0c;需要将 原 neo4j 数据导入到新的 tugraph 数据库中。预期走csv文件导入导出&#xff0c;但因为格式和数据库设计问题&#xff0c;操作起来比较麻烦&#xff08;可能是个人没…

Node.js安装(含npm安装vue-cli,安装element-ui)的详细配置

搭建前端框架 前端平台 量子计算机–10^5级别运算只需5min&#xff0c;这代表可以计算从宇宙大爆炸到现在的数据可以计算 安卓工程师–.xml node.js 下载 运行在win/linus的js——node.js 安装 建议不要动路径&#xff0c;可以避免很多问题&#xff0c;但是要保证C盘有至少1…

亚马逊云科技 re:Invent 2024重磅发布!Amazon Bedrock Data Automation 预览版震撼登场

AWS re:Invent 2024 已圆满落幕&#xff01; 在本次大会中&#xff0c;隆重推出了一项全新功能&#xff1a; Amazon Bedrock Data Automation&#xff08;预览版&#xff09;震撼登场&#xff01; New Amazon Bedrock capabilities enhance data processing and retrieval | …

JAVA:组合模式(Composite Pattern)的技术指南

1、简述 组合模式(Composite Pattern)是一种结构型设计模式,旨在将对象组合成树形结构以表示“部分-整体”的层次结构。它使客户端对单个对象和组合对象的使用具有一致性。 设计模式样例:https://gitee.com/lhdxhl/design-pattern-example.git 2、什么是组合模式 组合模式…

计算机基础 试题

建议做的时候复制粘贴,全部颜色改为黑色,做完了可以看博客对答案。 一、单项选择题(本大题共25小题,每小题2分,共50分〉 1.计算机内部采用二进制数表示信息,为了便于书写,常用十六进制数表示。一个二进制数0010011010110用十六进制数表示为 A.9A6 B.26B C.4D6 D.…

SAP ABAP-日期格式问题 SAP内部错误,反序列化JSON字符串时发生异常 值 20241215 不是根据 ABAP 的 XML 格式的有效日期

SAP ABAP-日期格式问题 SAP内部错误,反序列化JSON字符串时发生异常 值 20241215 不是根据 ABAP 的 XML 格式的有效日期 在SAP内部用 YYYYMMDD没有问题 外部传入参数

腾讯云云开发 Copilot 深度探索与实战分享

个人主页&#xff1a;♡喜欢做梦 欢迎 &#x1f44d;点赞 ➕关注 ❤️收藏 &#x1f4ac;评论 目录 一、引言 二、产品介绍 三、产品体验过程 四、整体总结 五、给开发者的复用建议 六、对 AI 辅助开发的前景展望 一、引言 在当今数字化转型加速的时代&#xff0c;…

中间件 redis安装

redis官网地址&#xff1a;Redis - The Real-time Data Platform 环境 CentOS Linux release 7.9.2009 (Core) java version "17.0.12" 2024-07-16 LTS 1、通过压缩包安装redis 1&#xff0c;远程下载redis压缩包&#xff0c;或去官网下载&#xff1a;Downloads …

CVE-2021-44228 漏洞复现

漏洞描述 什么是 log4j 和 log4j2 log4j 是 Apache 的一个开源日志库&#xff0c;是一个基于 Java 的日志记录框架&#xff0c;Log4j2 是 log4j 的后继者&#xff0c;其中引入了大量丰富的特性&#xff0c;可以控制日志信息输送的目的地为控制台、文件、GUI 组建等&#xff0…

SpringBoot02

1. 学习目标&#xff08;了解&#xff09; 2. Mybatis整合&数据访问&#xff08;操作&#xff09; 使用SpringBoot开发企业项目时&#xff0c;持久层数据访问是前端页面数据展示的基础&#xff0c;SpringBoot支持市面上常见的关系库产品(Oracle,Mysql,SqlServer,DB2等)对应…

答:C++需要学到什么程度再开始学 qt 比较合理?

有网友问&#xff1a;C需要学到什么程度再开始学 qt 比较合理&#xff1f; 南老师回答如下。 在我看来&#xff0c;这确实是一个好问题&#xff0c;但我的回答&#xff0c;大概很难成为一个好回答。 但我还是想回答&#xff0c;所以诚恳谢妖&#xff01; 如果有人问我&…

Elasticsearch8.17.0在mac上的安装

1、下载并安装 下载8.17版本es(目前最新版本)&#xff1a;Download Elasticsearch | Elastic 也可以通过历史版本列表页下载&#xff1a;Past Releases of Elastic Stack Software | Elastic 当然也可以指定具体版本号进行下载&#xff1a;Elasticsearch 8.17.0 | Elastic …

爬取Q房二手房房源信息

文章目录 1. 实战概述2. 网站页面分析3. 编写代码爬取Q房二手房房源信息3.1 创建项目与程序3.2 运行程序&#xff0c;查看结果 4. 实战小结 1. 实战概述 本次实战项目旨在通过编写Python爬虫程序&#xff0c;抓取深圳Q房网上的二手房房源信息。我们将分析网页结构&#xff0c;…