0302Prim算法-最小生成树-图-数据结构和算法(Java)

这篇具有很好参考价值的文章主要介绍了0302Prim算法-最小生成树-图-数据结构和算法(Java)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1 Prim算法

1.1 概述

1.1.1 算法描述

算法描述:

  1. 初始化最小生成树,只有一个起点;

  2. 每次将下一条连接树中顶点和其补集中顶点且权重最小的边(黑色表示)加入树中;

  3. 重复步骤中2,直至最小生成树中加入了V-1条边。

命题L。Prim算法能够得到任意加权连通图的最小生成树。

证明:有命题K可知,这颗不断生长的树定义了一个切分切不存在黑色的横切边。该算法会选取权重最小的横切边并根据贪心算法不断将它们标记为黑色。

  • 命题K在上一篇文章==0301概述-最小生成树-图-数据结构和算法(Java)==

最小生成树Prim算法如下图1.1.1-1所示:

0302Prim算法-最小生成树-图-数据结构和算法(Java)

1.1.2 数据结构

实现Prim算法需要用到一些常见的数据结构,我们会用以下方法表示树中的顶点、边和横切边。

  • 顶点:使用一个由顶点索引的布尔数组marked[],如果顶点v在树中,那么marked[v]的值为true;
  • 边;一条队列mst来保持最小生成树中的边;
  • 横切边:使用一条优先队列M in PQ<Edge>来根据权重比较所有边。
    • 优先队列可参考之前的文章==堆(二叉堆)-优先队列-数据结构和算法(Java)==和\02优先队列和索引优先队列-优先队列-数据结构和算法(Java)==

有了这些数据结构,我们可以回答“那条边的权重最小?”这个基本问题。

1.1.3 横切边集合维护
  • 每当我们向树中添加一条边之后,也向边中添加一个顶点;
  • 要维护包含所有横切边的集合,需要将连接这个顶点和其他所有不在树中的顶点边加入优先队列;
    • 用marked[]数组来识别这样的边;
  • 连接新加入树中顶点和其他已经在树中顶点 的所有边都失效了;
    • 这样的边已经不是横切边,因为它的两个顶点都在树中。
  • 将失效边从优先队列删除实现
    • 延时实现:将这些边先留在优先队列中,等到要删除它们的时候在检查边的有效性;
    • 即时实现:将这样的边从优先队列删除。

1.2 延时实现

1.2.1 实现代码

延时实现源代码1.2-1如下所示:

package edu.princeton.cs.algs4;

/**
 *  最小生成树Prim算法延时实现
 */
public class LazyPrimMST {
    private static final double FLOATING_POINT_EPSILON = 1E-12;

    /**
     * 最小生成树的总权重
    */
    private double weight;
  
    /**
     * 最小生成树中边的集合
    */
    private Queue<Edge> mst;
  
    /**
     * 图中顶点算法在树中标记
    */
    private boolean[] marked;
  
    /**
     * 维护横切边的优先队列
    */
    private MinPQ<Edge> pq;

    /**
     * 计算最小生成树
     * @param G 加权连通图
     */
    public LazyPrimMST(EdgeWeightedGraph G) {
        mst = new Queue<Edge>();
        pq = new MinPQ<Edge>();
        marked = new boolean[G.V()];
        // 遍历图中所有顶点
        for (int v = 0; v < G.V(); v++)
            if (!marked[v]) prim(G, v);

        // check optimality conditions
        assert check(G);
    }

    /**
     * 最小生成树Prim延时算法
    */
    private void prim(EdgeWeightedGraph G, int s) {
        scan(G, s);
        while (!pq.isEmpty()) {
            // 获取权重最小的边
            Edge e = pq.delMin();
            int v = e.either(), w = e.other(v);
            assert marked[v] || marked[w];
            // 如果边的两个顶点都在树中,跳过
            if (marked[v] && marked[w]) continue;
           // 该切分中权重最小横切边加入树中
            mst.enqueue(e);
            weight += e.weight();
            if (!marked[v]) scan(G, v);
            if (!marked[w]) scan(G, w);
        }
    }

    /**
     * 当前切分横切边加入优先队列
    */
    private void scan(EdgeWeightedGraph G, int v) {
        assert !marked[v];
        marked[v] = true;
        for (Edge e : G.adj(v))
            if (!marked[e.other(v)]) pq.insert(e);
    }
        
    /**
     * 最小生成树中所有边
     */
    public Iterable<Edge> edges() {
        return mst;
    }

    /**
     * 最小生成树权重之和
     */
    public double weight() {
        return weight;
    }

    /**
     * 校验加权连通图
    */
    private boolean check(EdgeWeightedGraph G) {

        // 校验权重
        double totalWeight = 0.0;
        for (Edge e : edges()) {
            totalWeight += e.weight();
        }
        if (Math.abs(totalWeight - weight()) > FLOATING_POINT_EPSILON) {
            System.err.printf("Weight of edges does not equal weight(): %f vs. %f\n", totalWeight, weight());
            return false;
        }

        // 校验是否有环
        UF uf = new UF(G.V());
        for (Edge e : edges()) {
            int v = e.either(), w = e.other(v);
            if (uf.find(v) == uf.find(w)) {
                System.err.println("Not a forest");
                return false;
            }
            uf.union(v, w);
        }

        // 校验是否是生成树森林
        for (Edge e : G.edges()) {
            int v = e.either(), w = e.other(v);
            if (uf.find(v) != uf.find(w)) {
                System.err.println("Not a spanning forest");
                return false;
            }
        }

        // check that it is a minimal spanning forest (cut optimality conditions)
        for (Edge e : edges()) {

            // all edges in MST except e
            uf = new UF(G.V());
            for (Edge f : mst) {
                int x = f.either(), y = f.other(x);
                if (f != e) uf.union(x, y);
            }

            // check that e is min weight edge in crossing cut
            for (Edge f : G.edges()) {
                int x = f.either(), y = f.other(x);
                if (uf.find(x) != uf.find(y)) {
                    if (f.weight() < e.weight()) {
                        System.err.println("Edge " + f + " violates cut optimality conditions");
                        return false;
                    }
                }
            }

        }

        return true;
    }
}

测试代码1.2.1-1如下所示:

public static void testLazyPrim() {
    String path = System.getProperty("user.dir") + File.separator + "asserts/tinyEWG.txt";
    In in = new In(path);
    EdgeWeightedGraph G = new EdgeWeightedGraph(in);
    LazyPrimMST mst = new LazyPrimMST(G);
    for (Edge e : mst.edges()) {
        StdOut.println(e);
    }
    StdOut.printf("%.5f\n", mst.weight());
}

测试结果:

0-7 0.16000
1-7 0.19000
0-2 0.26000
2-3 0.17000
5-7 0.28000
4-5 0.35000
6-2 0.40000
1.81000

测试用例Prim算法的轨迹图如下1.2.1所示:

0302Prim算法-最小生成树-图-数据结构和算法(Java)

1.2.2 性能分析

命题M。Prim算法的延时实现技术一幅含有V个顶点和E条边的加权连通图的最小生成树所需的空间与E成正比,所需的时间和 E log ⁡ E E\log E ElogE成正比(最快情况)。

证明:算法的瓶颈在与优先队列的insert()和delMin()方法中比较边的权重的次数。优先队列最多可能有E条边,这就是空间需求的上限。在最坏情况下,一次插入的成本为 ∽ lg ⁡ E \backsim \lg E lgE,删除最小元素 的成本 ∽ 2 lg ⁡ E \backsim 2\lg E 2lgE。因为最多只能插入E条边,删除E次最小元素,时间上限和 E log ⁡ E E\log E ElogE成正比。

1.3 即时实现

1.3.1 分析
  • 优化思想

要改进LazyPrimMST,可以从优先队列中删除失效边入手。

  • 我们关心的是连接树顶点和非树顶点中权重最小的边。

  • 当我们将顶点v加入到生成树中时,对于每个非树顶点w产生的变化是w到最小生成树的距离更近了。即我们不需要在优先队列中保存所有从w到树顶点的边,而只需要保存其中权重最小的边。

  • 我们通过遍历顶点v的连接表,可以实现。这样我们之后在优先队列中保存每个非树顶点w的一条边:当前将w与树中顶点连接起来的权重最小的那条边。

  • 结构更换:

    • 索引优先队列更换优先队列,因为我们需要访问优先队列中的元素。
    • 新增2个索引数组edgeTo[]和disTo[]
      • 如果顶点v不在树中但至少含有一条边和树相连,那么edgeTo[v]是将v和树连接的最短边(当前权重最小边),distTo[v]为这条边的权重。
      • 所有这类顶点v都保存在一条索引优先队列中,索引关联值是edgeTo[v]的边权重。
1.3.2 实现代码

Prim即使实现算法类PrimMST源代码如下:

package edu.princeton.cs.algs4;

/**
 *  最小生成树算法prim即时实现
 */
public class PrimMST {
    private static final double FLOATING_POINT_EPSILON = 1E-12;

	  /**
     * 顶点和树连接的最短边
     */
    private Edge[] edgeTo;
    
     /**
     * 最短边对应的权重
     */
    private double[] distTo;
    
     /**
     * 顶点是否在树中
     */
    private boolean[] marked;     
    
     /**
     * 维护横切边集合
     */
    private IndexMinPQ<Double> pq;

    /**
     * Compute a minimum spanning tree (or forest) of an edge-weighted graph.
     * @param G the edge-weighted graph
     */
    public PrimMST(EdgeWeightedGraph G) {
        edgeTo = new Edge[G.V()];
        distTo = new double[G.V()];
        marked = new boolean[G.V()];
        pq = new IndexMinPQ<Double>(G.V());
        for (int v = 0; v < G.V(); v++)
            distTo[v] = Double.POSITIVE_INFINITY;

        for (int v = 0; v < G.V(); v++)      // run from each vertex to find
            if (!marked[v]) prim(G, v);      // minimum spanning forest

        // check optimality conditions
        assert check(G);
    }

    /**
     * prim算法即时实现
     */
    private void prim(EdgeWeightedGraph G, int s) {
        distTo[s] = 0.0;
        pq.insert(s, distTo[s]);
        while (!pq.isEmpty()) {
            int v = pq.delMin();
            scan(G, v);
        }
    }

    /**
     * 扫描顶点v的连接表,使优先队列中只保留非树顶点连接树顶点最小边
     */
    private void scan(EdgeWeightedGraph G, int v) {
        marked[v] = true;
        for (Edge e : G.adj(v)) {
            int w = e.other(v);
            if (marked[w]) continue;         // v-w is obsolete edge
            if (e.weight() < distTo[w]) {
                distTo[w] = e.weight();
                edgeTo[w] = e;
                if (pq.contains(w)) pq.decreaseKey(w, distTo[w]);
                else                pq.insert(w, distTo[w]);
            }
        }
    }

    /**
     * 最小生成树中边
     */
    public Iterable<Edge> edges() {
        Queue<Edge> mst = new Queue<Edge>();
        for (int v = 0; v < edgeTo.length; v++) {
            Edge e = edgeTo[v];
            if (e != null) {
                mst.enqueue(e);
            }
        }
        return mst;
    }

    /**
     * 生成树权重
     */
    public double weight() {
        double weight = 0.0;
        for (Edge e : edges())
            weight += e.weight();
        return weight;
    }


    // check optimality conditions (takes time proportional to E V lg* V)
    private boolean check(EdgeWeightedGraph G) {

        // check weight
        double totalWeight = 0.0;
        for (Edge e : edges()) {
            totalWeight += e.weight();
        }
        if (Math.abs(totalWeight - weight()) > FLOATING_POINT_EPSILON) {
            System.err.printf("Weight of edges does not equal weight(): %f vs. %f\n", totalWeight, weight());
            return false;
        }

        // check that it is acyclic
        UF uf = new UF(G.V());
        for (Edge e : edges()) {
            int v = e.either(), w = e.other(v);
            if (uf.find(v) == uf.find(w)) {
                System.err.println("Not a forest");
                return false;
            }
            uf.union(v, w);
        }

        // check that it is a spanning forest
        for (Edge e : G.edges()) {
            int v = e.either(), w = e.other(v);
            if (uf.find(v) != uf.find(w)) {
                System.err.println("Not a spanning forest");
                return false;
            }
        }

        // check that it is a minimal spanning forest (cut optimality conditions)
        for (Edge e : edges()) {

            // all edges in MST except e
            uf = new UF(G.V());
            for (Edge f : edges()) {
                int x = f.either(), y = f.other(x);
                if (f != e) uf.union(x, y);
            }

            // check that e is min weight edge in crossing cut
            for (Edge f : G.edges()) {
                int x = f.either(), y = f.other(x);
                if (uf.find(x) != uf.find(y)) {
                    if (f.weight() < e.weight()) {
                        System.err.println("Edge " + f + " violates cut optimality conditions");
                        return false;
                    }
                }
            }

        }

        return true;
    }
}

主要算法:

  • 将顶点v加入树中后,遍历v的邻接表(边)

  • 邻接边对应另外一个顶点w,如果已经在树中,跳过;

  • 没在树中,判断非树顶点w与树中顶点v连接边权重是否小于之前记录非树顶点w与树中顶点连接边的权重

    • 是,判断索引优先队列算法包含索引w
      • 是更新,明确权重比原先的小只需要更新索引对应的权重值后上浮即可;
      • 不是新加入

测试代码如下1.3.2-2所示:

    public static void testPrim() {
        String path = System.getProperty("user.dir") + File.separator + "asserts/tinyEWG.txt";
        In in = new In(path);
        EdgeWeightedGraph G = new EdgeWeightedGraph(in);
        PrimMST mst = new PrimMST(G);
        for (Edge e : mst.edges()) {
            StdOut.println(e);
        }
        StdOut.printf("%.5f\n", mst.weight());
    }

测试结果:

1-7 0.19000
0-2 0.26000
2-3 0.17000
4-5 0.35000
5-7 0.28000
6-2 0.40000
0-7 0.16000
1.81000
1.3.3 性能分析

命题N。Prim算法的即时实现即时一幅含有V个顶点和E条边的连通加权无向图的最小生成树所需的空间和V成正比,所需时间和 E log ⁡ V E\log V ElogV成正比(最坏情况)。

证明:因为优先队列中的顶点时最多为V,且使用3条由递给你的所有的数组,所以所需空间上限和V成正比。算法会进行V次插入操作,V次删除最小元素操作和(在最坏情况下)E次改变优先级操作。已知在基于堆实现索引优先队列中所有这些操作的增长量级为 log ⁡ V \log V logV,所以将所有这些加起来可知算法所需时间和 E log ⁡ V E\log V ElogV成正比。

结语

如果小伙伴什么问题或者指教,欢迎交流。

❓QQ:806797785

⭐️源代码仓库地址:https://gitee.com/gaogzhen/algorithm

参考链接:

[1][美]Robert Sedgewich,[美]Kevin Wayne著;谢路云译.算法:第4版[M].北京:人民邮电出版社,2012.10.p398-404.文章来源地址https://www.toymoban.com/news/detail-415294.html

到了这里,关于0302Prim算法-最小生成树-图-数据结构和算法(Java)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 数据结构与算法课程设计---最小生成树的应用

    1.问题 假定有这么一个问题,有11个城市,城市之间有一些天然气管道,铺设天然气管道需要花费不同的金额,现在要你选择其中一些天然气管道,使得所有城市可以互相联通且花费最小。 2.分析 我们把这个问题抽象为一张图,每个城市是一个顶点,城市与城市之间的管道是

    2024年02月08日
    浏览(36)
  • 【数据结构与算法】图的搜索——广度优先遍历、最小生成树

    邻接链表: 用字典实现.有向图的邻接链表的总长度等于图的边个数;无向图的邻接链表的总长度等于图的边个数的2倍. 邻接矩阵:用于最短路径算法. 该数据结构维护一个不相交动态集的集合,每个集合有一个代表,不关心谁做代表。 支持三种操作: MAKE_SET(x) FIND_SET(x) UNION(x,y

    2024年02月19日
    浏览(32)
  • Java高阶数据结构 & 并查集 & 最小生成树

    并查集与最小生成树 1.1 并查集的原理 在一些应用问题中,我们常常会遇到一类问题 一开始是一个人 后来新增的人可能与这个人有关系,也可能与这个人无关系。 一个人与一个人有关系,这个人与另一个人也有关系,那么三人都有关系。 有关系的和没关系的之间是不同的类

    2024年02月03日
    浏览(28)
  • 【数据结构】无向图的最小生成树(Prime,Kruskal算法)

    连通图 :在 无向图 中,若从顶点v1到顶点v2有路径,则称顶点v1与顶点v2是连通的。如果图中 任意一对顶点都是连通的 ,则称此图为连通图 强连通图 :在 有向图 中,若在每一对顶点vi和vj之间都存在一条从vi到vj的路径,也存在一条从vj到vi的路径,则称此图是强连通图 生成

    2024年01月24日
    浏览(32)
  • 数据结构---最小生成树((普里姆算法)C语言看了就懂教程)

        普里姆算法就是“加点法”,是一种将连通网转换成最小生成树的一种算法     在一个连通图的所有生成树中,各边代价之和最小的那颗生成树称为该连通图的最小代价生成树(MST)     ①对于任意一张连通图,假设 N = (V,E)是连通网,TE就是最小生成树中边的集合    

    2024年02月03日
    浏览(28)
  • 考研算法复试刷题19天:Prim算法求最小生成树 【prim,最小生成树】

    参考博客:图解:什么是最小生成树? - 知乎 (zhihu.com)  总结下来的过程就是,一张图,我们将他化为树的形式,也就是生成树。那么最小生成树有是啥呢? 所谓一个 带权图 的最小生成树,就是原图中边的权值最小的生成树 ,所谓最小是指边的权值之和小于或者等于其它

    2024年02月07日
    浏览(45)
  • 数据结构——普里姆(Prim)算法

    普里姆算法(Prim算法),图论中的一种算法,可在加权连通图里搜索最小生成树。意即由此算法搜索到的边子集所构成的树中,不但包括了连通图里的所有顶点,且其所有边的权值之和亦为最小。 以下是数据结构中关于普里姆算法的操作(编程风格参考严蔚敏版数据结构)

    2024年02月06日
    浏览(33)
  • (数据结构)普利姆算法(Prim)实现

    假设要在n个城市之间建立通信联络网,每两个城市之间建立线路都需要花费不同大小的经费,则连通n个城市只需要n-1个条线路,最小生成树解决的问题就是: 如何在最节省经费的前提下建立这个通信网 也可以理解为:n个城市之间最多可以建立n(n-1)/2条线路,我们要从这里挑

    2024年02月03日
    浏览(33)
  • 贪心算法:最小生成树Prim算法

    👨‍🎓作者简介:一位喜欢写作,计科专业大二菜鸟 🏡个人主页:starry陆离 🕒首发日期:2022年5月31日星期二 🌌上期文章:动态规划:多重背包问题 📚订阅专栏:算法分析与设计 如果文章有帮到你的话记得点赞👍+收藏💗支持一下哦 这是大一暑假的c笔记,再一次写pri

    2024年02月01日
    浏览(34)
  • 最小生成树—Prim算法

    我们要讨论的问题是如何在一个 无向图 中找到它的最小生成树,虽然这个问题对有向图也有意义,但是处理起来更麻烦。 一个无向图 G 的最小生成树就是连接 G 上所有顶点的边构成的树,且这些边的总权值最低。当且仅当图是 连通的 才有最小生成树。 在无向图的最小生成

    2024年02月09日
    浏览(31)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包