【算法基础实验】图论-最小生成树Prim的延迟实现

devtools/2024/9/22 19:21:57/

最小生成树-Prim的延迟实现

理论基础

树的基本性质

用一条边连接树中的任意两个顶点都会产生一个新的环;
从树中删去一条边将会得到两棵独立的树。

切分定理的定义

定义。图的一种切分是将图的所有顶点分为两个非空且不重叠的两个集合。横切边
是一条连接两个属于不同集合的顶点的边。
命题 (切分定理)。在一幅加权图中,给定任意的切分,它的横切边中的权重最
小者必然属于图的最小生成树。

切分定理的证明

请添加图片描述

证明。令e为权重最小的横切边,T为图的最小生成树。我们采用反证法:假设T
不包含e。那么如果将e加入T,得到的图必然含有一条经过e的环,且这个环
至少含有另一条横切边——设为f,f的权重必然大于e(因为e的权重是最小的
且图中所有边的权重均不同)。那么我们删掉f而保留e就可以得到一棵权重更
小的生成树。这和我们的假设T矛盾。

实验数据和算法流程

请添加图片描述

请添加图片描述

代码实现

java">import edu.princeton.cs.algs4.In;
import edu.princeton.cs.algs4.myMinPQ;
import edu.princeton.cs.algs4.StdOut;public class myLazyPrimMST {private boolean[] marked;private myLinkedQueue<myEdge> mst;private myMinPQ<myEdge> pq;private double totalWeight;public myLazyPrimMST(myEdgeWeightedGraph G){marked = new boolean[G.V()];mst = new myLinkedQueue<myEdge>();pq = new myMinPQ<myEdge>();visit(G,0);while(!pq.isEmpty()){myEdge e = pq.delMin();int v=e.either();int w=e.other(v);if(marked[v]&&marked[w]) continue;mst.enqueue(e);if(!marked[v]) visit(G,v);if(!marked[w]) visit(G,w);}}private void visit(myEdgeWeightedGraph G, int v){marked[v] = true;for(myEdge e:G.adj(v))if(!marked[e.other(v)])pq.insert(e);}public Iterable<myEdge> edges(){ return mst; }public double weight(){totalWeight = 0.0;for(myEdge e:edges())totalWeight +=e.weight();return totalWeight;}public static void main(String[] args){In in = new In(args[0]);myEdgeWeightedGraph G = new myEdgeWeightedGraph(in);myLazyPrimMST mst = new myLazyPrimMST(G);for(myEdge e:mst.edges())StdOut.println(e);StdOut.print(mst.weight());}
}

代码详解

这段代码实现了一个名为 myLazyPrimMST 的类,用于计算加权无向图的最小生成树(MST),采用的是 Prim 的延迟算法。下面是代码的主要功能和操作步骤的详细解释:

  1. 类变量
    • private boolean[] marked;:标记数组,用于记录图的顶点是否已被访问。
    • private myLinkedQueue<myEdge> mst;:用于存储构成最小生成树的边。
    • private myMinPQ<myEdge> pq;:优先队列,用于按边的权重顺序访问边。
    • private double totalWeight;:记录最小生成树的总权重。
  2. 构造函数
    • 在构造最小生成树时,首先初始化标记数组、边队列和优先队列。
    • 从顶点0开始访问图,将其相邻的未访问边加入优先队列。
    • 循环从优先队列中取出最小边,如果这条边的两个顶点已经被标记,则忽略此边;否则,将其加入到生成树中,并访问相邻的未标记顶点。
  3. 访问顶点(visit 方法)
    • 标记顶点为已访问。
    • 遍历与顶点相连的所有边,如果边的另一端未被标记,则将该边加入优先队列。
  4. 获取生成树的边和权重
    • edges() 方法返回构成最小生成树的边。
    • weight() 方法计算最小生成树的总权重,通过遍历所有生成树的边并累加它们的权重。
  5. 主函数
    • 从文件读取图的数据构造图对象。
    • 创建 myLazyPrimMST 对象来生成最小生成树。
    • 输出生成树的所有边和总权重。

通过这种方式,myLazyPrimMST 类使用延迟的 Prim 算法有效地找到了一个给定图的最小生成树,这种算法特别适合处理稀疏图。

实验

代码编译

java">$ javac myLazyPrimMST.java

代码运行

代码运行输出的是计算得到的最小生成树的所有边,以及最小生成树所有边的总权重

java">$ java myLazyPrimMST ..\data\tinyEWG.txt
0-7 0.16
1-7 0.19
0-2 0.26
2-3 0.17
5-7 0.28
4-5 0.35
6-2 0.40
1.81

参考资料

算法(第四版) 人民邮电出版社


http://www.ppmy.cn/devtools/30263.html

相关文章

Java面试题:Spring里面的@RestController和@ResponseBody有什么作用?

ResponseBody ResponseBody一般是加在方法上&#xff0c;将返回的对象解析成xml或者json&#xff0c;返回给请求的调用者。一般是用于服务之间的调用&#xff0c;或者前端请求后端时&#xff0c;使用ajax请求。 如果不加ResponseBody&#xff0c;一般就是返回的url&#xff0c…

2024-04学习笔记

1.sql优化-子查询改为外连接 1.改之前 改之前是这样&#xff0c;那针对查出来的每一条数据&#xff0c;都要执行一次箭头所指的函数 执行的sql很慢 2.改之后 改之后是这样&#xff0c;整体做外连接&#xff0c;不用每一条都再执行一次查询 执行时间缩短了好几倍 2.Mybatis中…

农村公交与异构无人机协同配送优化

针对农村公交与异构无人机协同配送的优化问题,可以从以下几个方面进行探讨: 1. 融合公交与无人机配送 公交物流体系:利用农村公交网络,建立以公交车辆为基础的物流配送体系。公交车辆可以沿途收集或投递货物,提高物流配送效率。无人机辅助配送:在公交物流体系的基础上,…

Es6和Es5的区别?

ES5和ES6都是JavaScript语言的版本&#xff0c;ES5在2009年发布&#xff0c;ES6在2015年发布&#xff0c;两者之间有以下的区别&#xff1a; 1、变量声明方式不同&#xff1a;ES5使用var关键字进行变量声明&#xff0c;而ES6则引入了let和const关键字来声明变量。 2、块级作用…

服务运营 | 精选:花钱买开心!体验型服务设计中的调度优化

编者按 在体验经济时代&#xff0c;企业逐渐从提供产品转变为提供体验&#xff0c;只有了解顾客的行为&#xff0c;才能对服务进行更好的设计&#xff0c;从而提高顾客的体验和忠诚度&#xff0c;实现企业与顾客的双赢。如何优化顾客体验便是体验型服务设计&#xff08;Experie…

在Mac上恢复已删除文件夹的最佳方法

“嗨&#xff0c;我从我的Mac Documents文件夹中删除了很多文件夹。已删除的文件夹包含我的重要文档和文件&#xff0c;是否可以取回它们&#xff1f;垃圾桶已被清洁软件清空。如何在我的Mac上恢复已删除的文件夹&#xff1f; 当您在 Mac 上删除 1 或 2 个文件夹时&#xff0c…

【强训笔记】day7

NO.1 思路&#xff1a;双指针模拟&#xff0c;begin表示最长数字字符串最后一个字符&#xff0c;而len表示数字字符串的长度&#xff0c;i用来遍历&#xff0c;如果为数字&#xff0c;那么定义j变量继续遍历&#xff0c;直到不为数字&#xff0c;i-j如果大于len&#xff0c;就…

RSA实现中弱密钥漏洞分析

RSA实现中弱密钥漏洞分析 “Analyzing Weak Key Vulnerabilities in RSA Implementation” 完整下载链接:RSA实现中弱密钥漏洞分析 文章目录 RSA实现中弱密钥漏洞分析摘要第一章 引言1.1 研究背景1.2 研究目的1.3 研究意义 第二章 RSA算法基础2.1 RSA算法原理2.2 RSA密钥生成…