Java中Prime算法的原理是什么与怎么实现

Prim算法介绍

1.点睛

在生成树的过程中,把已经在生成树中的节点看作一个集合,把剩下的节点看作另外一个集合,从连接两个集合的边中选择一条权值最小的边即可。

2.算法介绍

Java中Prime算法的原理是什么与怎么实现

首先任选一个节点,例如节点1,把它放在集合 U 中,U={1},那么剩下的节点为 V-U={2,3,4,5,6,7},集合 V 是图的所有节点集合。

Java中Prime算法的原理是什么与怎么实现

现在只需要看看连接两个集合(U 和 V-U)的边中,哪一条边的权值最小,把权值最小的边关联的节点加入集合 U 中。从上图可以看出,连接两个集合的 3 条边中,1-2 边的权值最小,选中它,把节点 2 加入集合 U 中,U={1,2},V – U={3,4,5,6},如下图所示。

Java中Prime算法的原理是什么与怎么实现

再从连接两个集合(U 和 V-U)的边中选择一条权最小的边。从上图看出,在连接两个集合的4条边中,节点2到节点7的边权值最小,选中这条边,把节点7加入集合U={1,2,7}中,V-U={3,4,5,6}。

如此下去,直到 U=V 结束,选中的边和所有的节点组成的图就是最小生成树。这就是 Prim 算法。

直观地看图,很容易找出集合 U 到 集合 U-V 的边中哪条边的权值是最小的,但在程序中穷举这些边,再找最小值,则时间复杂度太高。可以通过设置数组巧妙解决这个问题,closet[j] 表示集合 V-U 中的节点 j 到集合 U 中的最邻近点,lowcost[j] 表示集合 V-U 中节点 j 到集合 U 中最邻近点的边值,即边(j,closest[j]) 的权值。

例如在上图中,节点 7 到集合 U 中的最邻近点是2,cloeest[7]=2。节点 7 到最邻近点2 的边值为1,即边(2,7)的权值,记为 lowcost[7]=1,如下图所示。

Java中Prime算法的原理是什么与怎么实现

所以只需在集合 V – U 中找到 lowcost[] 只最小的节点即可。

3. 算法步骤

1.初始化

令集合 U={u0},u0 属于 V,并初始化数组 closest[]、lowcost[]和s[]。

2.在集合 V-U 中找 lowcost 值最小的节点t,即 lowcost[t]=min{lowcost[j]},j 属于 V-U,满足该公式的节点 t 就是集合 V-U 中连接 U 的最邻近点。

3.将节点 t 加入集合 U 中。

4.如果集合 V – U 为空,则算法结束,否则转向步骤 5。

5.对集合 V-U 中的所有节点 j 都更新其 lowcost[] 和 closest[]。if(C[t][j]<lowcost[j]){lowcost[j]=C[t][j];closest[j]=t;},转向步骤2。

按照上面步骤,最终可以得到一棵权值之和最小的生成树。

4.图解

图 G=(V,E)是一个无向连通带权图,如下图所示。

Java中Prime算法的原理是什么与怎么实现

1 初始化。假设 u0=1,令集合 U={1},集合 V-U={2,3,4,5,6,7},s[1]=true,初始化数组 closest[]:除了节点1,其余节点均为1,表示集合 V-U 中的节点到集合 U 的最邻近点均为1.lowcost[]:节点1到集合 V-U 中节点的边值。closest[] 和 lowcost[] 如下图所示。

Java中Prime算法的原理是什么与怎么实现

初始化后的图为:

Java中Prime算法的原理是什么与怎么实现

2 找 lowcost 最小的节点,对应的 t=2,选中的边和节点如下图。

Java中Prime算法的原理是什么与怎么实现

3 加入集合U中。将节点 t 加入集合 U 中,U={1,2},同时更新 V-U={3,4,5,6,7}

4 更新。对 t 在集合 V-U 中的每一个邻接点 j,都可以借助 t 更新。节点 2 的邻接点是节点 3 和节点7。

C[2][3]=20<lowcost[3]=无穷大,更新最邻近距离 lowcost[3]=20,最邻近点 closest[3]=2;

C[2][7]=1<lowcost[7]=36,更新最邻近距离 lowcost[7]=1,最邻近点 closest[7]=2;

更新后的 closest[] 和 lowcost[] 如下图所示。

Java中Prime算法的原理是什么与怎么实现

更新后的集合如下图所示:

Java中Prime算法的原理是什么与怎么实现

5 找 lowcost 最小的节点,对应的 t=7,选中的边和节点如下图。

Java中Prime算法的原理是什么与怎么实现

6 加入集合U中。将节点 t 加入集合 U 中,U={1,2,7},同时更新 V-U={3,4,5,6}

7 更新。对 t 在集合 V-U 中的每一个邻接点 j,都可以借助 t 更新。节点 7 的邻接点是节点 3、4、5、6。

  • C[7][3]=4<lowcost[3]=20,更新最邻近距离 lowcost[3]=4,最邻近点 closest[3]=7;

  • C[7][4]=4<lowcost[4]=无穷大,更新最邻近距离 lowcost[3]=9,最邻近点 closest[4]=7;

  • C[7][5]=4<lowcost[5]=无穷大,更新最邻近距离 lowcost[3]=16,最邻近点 closest[5]=7;

  • C[7][6]=4<lowcost[6]=28,更新最邻近距离 lowcost[3]=25,最邻近点 closest[6]=7;

更新后的 closest[] 和 lowcost[] 如下图所示。

Java中Prime算法的原理是什么与怎么实现

更新后的集合如下图所示:

Java中Prime算法的原理是什么与怎么实现

8 找 lowcost 最小的节点,对应的 t=3,选中的边和节点如下图。

Java中Prime算法的原理是什么与怎么实现

9 加入集合U中。将节点 t 加入集合 U 中,U={1,2,3,7},同时更新 V-U={4,5,6}

10 更新。对 t 在集合 V-U 中的每一个邻接点 j,都可以借助 t 更新。节点 3 的邻接点是节点 4。

C[3][4]=15>lowcost[4]=9,不更新

closest[] 和 lowcost[] 数组不改变。

更新后的集合如下图所示:

Java中Prime算法的原理是什么与怎么实现

11 找 lowcost 最小的节点,对应的 t=4,选中的边和节点如下图。

Java中Prime算法的原理是什么与怎么实现

12 加入集合U中。将节点 t 加入集合 U 中,U={1,2,3,4,7},同时更新 V-U={5,6}

13 更新。对 t 在集合 V-U 中的每一个邻接点 j,都可以借助 t 更新。节点 4 的邻接点是节点 5。

C[4][5]=3<lowcost[5]=16,更新最邻近距离 lowcost[5]=3,最邻近点 closest[5]=4;

更新后的 closest[] 和 lowcost[] 如下图所示。

Java中Prime算法的原理是什么与怎么实现

更新后的集合如下图所示:

Java中Prime算法的原理是什么与怎么实现

14 找 lowcost 最小的节点,对应的 t=5,选中的边和节点如下图。

Java中Prime算法的原理是什么与怎么实现

15 加入集合U中。将节点 t 加入集合 U 中,U={1,2,3,4,5,7},同时更新 V-U={6}

16 更新。对 t 在集合 V-U 中的每一个邻接点 j,都可以借助 t 更新。节点 5 的邻接点是节点 6。

C[5][6]=17<lowcost[6]=25,更新最邻近距离 lowcost[6]=17,最邻近点 closest[6]=5;

更新后的集合如下图所示:

Java中Prime算法的原理是什么与怎么实现

17 找 lowcost 最小的节点,对应的 t=6,选中的边和节点如下图。

Java中Prime算法的原理是什么与怎么实现

18 加入集合U中。将节点 t 加入集合 U 中,U={1,2,3,4,5,6,7},同时更新 V-U={}

19 更新。对 t 在集合 V-U 中的每一个邻接点 j,都可以借助 t 更新。节点 6 在集合 V-U 中无邻接点。不用更新 closest[] 和 lowcost[] 。

20 得到的最小生成树如下。最小生成树的权值之和为 57.

Java中Prime算法的原理是什么与怎么实现

Prime 算法实现

1.构建后的图

Java中Prime算法的原理是什么与怎么实现

2.代码

package graph.prim; import java.util.Scanner; public class Prim {    static final int INF = 0x3f3f3f3f;    static final int N = 100;    // 如果s[i]=true,说明顶点i已加入U    static boolean s[] = new boolean[N];    static int c[][] = new int[N][N];    static int closest[] = new int[N];    static int lowcost[] = new int[N];     static void Prim(int n) {        // 初始时,集合中 U 只有一个元素,即顶点 1        s[1] = true;        for (int i = 1; i <= n; i++) {            if (i != 1) {                lowcost[i] = c[1][i];                closest[i] = 1;                s[i] = false;            } else                lowcost[i] = 0;        }        for (int i = 1; i < n; i++) {            int temp = INF;            int t = 1;            // 在集合中 V-u 中寻找距离集合U最近的顶点t            for (int j = 1; j <= n; j++) {                if (!s[j] && lowcost[j] < temp) {                    t = j;                    temp = lowcost[j];                }            }            if (t == 1)                break; // 找不到 t,跳出循环            s[t] = true; // 否则,t 加入集合U            for (int j = 1; j <= n; j++) { // 更新 lowcost 和 closest                if (!s[j] && c[t][j] < lowcost[j]) {                    lowcost[j] = c[t][j];                    closest[j] = t;                }            }        }    }     public static void main(String[] args) {        int n, m, u, v, w;        Scanner scanner = new Scanner(System.in);        n = scanner.nextInt();        m = scanner.nextInt();        int sumcost = 0;        for (int i = 1; i <= n; i++)            for (int j = 1; j <= n; j++)                c[i][j] = INF;        for (int i = 1; i <= m; i++) {            u = scanner.nextInt();            v = scanner.nextInt();            w = scanner.nextInt();            c[u][v] = c[v][u] = w;        }        Prim(n);        System.out.println("数组lowcost:");         for (int i = 1; i <= n; i++)            System.out.print(lowcost[i] + " ");         System.out.println();        for (int i = 1; i <= n; i++)            sumcost += lowcost[i];        System.out.println("最小的花费:" + sumcost);    }}

3.测试

Java中Prime算法的原理是什么与怎么实现

到此,相信大家对“Java中Prime算法的原理是什么与怎么实现”有了更深的了解,不妨来实际操作一番吧!这里是亿速云网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

文章标题:Java中Prime算法的原理是什么与怎么实现,发布者:亿速云,转载请注明出处:https://worktile.com/kb/p/25230

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
亿速云的头像亿速云认证作者
上一篇 2022年9月15日 上午1:51
下一篇 2022年9月15日 上午1:53

相关推荐

  • 如何分析SQLMap和SQLi注入防御

    名列前茅部分:Sqlmap使用 1.1 sqlmap介绍 1. 前边说了一些sql注入的基础语句,但是手工注入很麻烦,我们可以借助sqlmap这个强大的sql注入工具,进行数据的获取. 2. sqlmap介绍 (1)#sqlmap是一种开源的渗透测试工具,可以自动检测和利用SQL注入漏洞以及接入该数…

    2022年9月16日
    69800
  • jquery怎么获取屏幕宽高

    具体方法如下: 1.获取当前窗口可视区域宽度 $(window).width(); 2.获取当前窗口可视区域高度 $(window).height(); 3.获取当前窗口文档对象宽度 $(document).width(); 4.获取当前窗口文档的高度 $(document).height(); 相…

    2022年8月29日
    57100
  • windows驱动精灵屏保如何关闭

    驱动精灵屏保关闭方法: 1、打开驱动精灵进入主页点击“百宝箱”。 2、在“其他推荐”中找到“护眼模式”。 3、将“定时休息”边上的开关关闭即可。 感谢各位的阅读,以上就是“windows驱动精灵屏保如何关闭”的内容了,经过本文的学习后,相信大家对windows驱动精灵屏保如何关闭这一问题有了更深刻的…

    2022年9月13日
    89400
  • Pytorch中的tensor数据结构实例代码分析

    torch.Tensor torch.Tensor 是一种包含单一数据类型元素的多维矩阵,类似于 numpy 的 array。Tensor 可以使用 torch.tensor() 转换 Python 的 list 或序列数据生成,生成的是dtype 默认是 torch.FloatTensor。 注意…

    2022年9月15日
    82900
  • php索引超出了数组界限如何解决

    索引超出了数组界限解决方法: 1、在test.php文件内使用header设置test.php执行的编码为utf8,避免乱码的出现。 2、随后在test.php文件内创建一个测试数组,对应的索引值分别为0、4、8。 3、在test.php文件中使用array_values方法将上一步的数据重新排序,…

    2022年9月10日
    77500
  • 如何进行Apache Commons Collections反序列化漏洞分析与复现

    1.1 状态 完成漏洞挖掘条件分析、漏洞复现。 1.2 漏洞分析 存在安全缺陷的版本:Apache Commons Collections3.2.1以下,【JDK版本:1.7.0_80】Apache Maven 3.6.3。 POC核心代码: package com.patrilic.vul;imp…

    2022年9月8日
    89800
  • JavaScript变量or循环中的var和let怎么使用

    在for循环中使用var声明初始化带来的问题 // 一道经典面试题:var funcs = [];for (var i = 0; i < 3; i++) { funcs[i] = function() { console.log(“My value: ” + i) };}for (var j …

    2022年9月15日
    41100
  • php中有哪些字符串定界符

    php字符串定界符有两种:1、heredoc定界符,在“<<<”运算符之后要提供一个标识符,然后换行,接下来是字符串本身,最后要用前面定义的标识符作为结束标志。2、nowdoc定界符,在“<<<”运算符之后要提供一个被单引号括起来的标识符,然后换行,接下来是字符串…

    2022年9月26日
    86000
  • 如何用rank函数排名

    用rank函数排名的方法: 1、首先进入空白单元格,然后输入:=RANK( )来进行排名。 2、然后去点击E2单元格,即可获得rank的名列前茅个参数,然后输入逗号。 3、之后就可以去框选所有的数据,并且去获取函数的第二个参数。 4、然后按下F4,即可对rank函数的第二个参数做绝对引用。 5、按下…

    2022年8月30日
    82000
  • cad字体不显示数字怎么解决

    解决方法 1、首先我们打开软件,找到菜单栏上的格式,再找到下拉菜单上的标注样式管理器。 2、然后在跳转出来的对话框中,点击当前使用的标注样式,然后再点击右侧的修改。 3、然后在跳转的新页面里,我们点击文字的选项,在文字高度这一块把高度数值改大一些。 以上就是“cad字体不显示数字怎么解决”这篇文章的…

    2022年9月16日
    71800
注册PingCode 在线客服
站长微信
站长微信
电话联系

400-800-1024

工作日9:30-21:00在线

分享本页
返回顶部