最小生成树之Kruskal算法和Prim算法

  Kruskal多用于稀疏图,prim多用于稠密图。

  根据图的深度优先遍历和广度优先遍历,可以用最少的边连接所有的顶点,而且不会形成回路。这种连接所有顶点并且路径唯一的树型结构称为生成树或扩展树。实际中,希望产生的生成树的所有边的权值和最小,称之为最小生成树。常见的最小生成树算法有Kruskal算法和Prim算法。

Kruskal算法

n个顶点的图最小生成树步骤如下:

1、边的权值升序排序;

2、选取所有未遍历的边中权值最小的边,判断加入后是否形成回路,若形成回路,放弃之,重新从未被遍历的边中选择。

3、重复上述步骤,直到选中n-1条边。

代码(并查集+kruskal):

#include <iostream>
#include <algorithm>
using namespace std; #define MAX 105 int N, E, Answer;
int u[MAX], v[MAX], w[MAX];
int p[MAX], r[MAX]; int cmp(const int i, const int j)
{
return w[i] < w[j];
} int find(int x)
{
return (p[x] == x) ? x : p[x] = find(p[x]);
} void kruskal()
{
int cnt = ;
int x, y; for(int i = ; i < N; i++) p[i] = i;
for(int i = ; i < E; i++) r[i] = i; sort(r, r+E, cmp); //升序排列 for(int i = ; i < E; i++)
{
int e = r[i];
x = find(u[e]); //并查集
y = find(v[e]);
if(x != y) //祖先不相同,则不会形成环
{
Answer += w[e];
p[x] = y;
cnt++;
}
}
if(cnt < N - ) Answer = ;
} int main()
{
while(cin >> N >> E, N)
{
Answer = ;
for(int i = ; i < E; i++)
{
cin >> u[i] >> v[i] >> w[i];
} kruskal(); cout << Answer << endl;
}
}

输入:


Prim算法

算法描述

普利姆算法求最小生成树时候,和边数无关,只和定点的数量相关,所以适合求稠密网的最小生成树,时间复杂度为O(n*n)。

算法过程:

1.将一个图的顶点分为两部分,一部分是最小生成树中的结点(A集合),另一部分是未处理的结点(B集合)。

2.首先选择一个结点,将这个结点加入A中,然后,对集合A中的顶点遍历,找出A中顶点关联的边权值最小的那个(设为v),将此顶点从B中删除,加入集合A中。

3.递归重复步骤2,直到B集合中的结点为空,结束此过程。

4.A集合中的结点就是由Prime算法得到的最小生成树的结点,依照步骤2的结点连接这些顶点,得到的就是这个图的最小生成树。

算法实现具体过程:

1.将第一个点放入最小生成树的集合中(标记visited[i]=1意思就是最小生成树集合)。

2.从第二个点开始,初始化dist[i]为跟1点相连(仅仅相连)的边的权值(dist[i]不是这个点的最小权值!在以后会逐步更新)。

3.找最小权值的边。从第二点开始遍历,如果不是最小生成树的集合的点,则找出从2到n的最小权值(dist[j])。

4.将找出来的最小权值的边的顶点加入最小生成树的集合中(标记visited[i] = 1),权值想加。

5.更新dist[j]集合。假设第一次:dist[2]代表与1相连的点的权值,现在加入了index点。则比较index点与2点的边graph[index][2]和dist[2]的大小,若 dist[2]大,则dist[2] = graph[index][2]。(关键步骤:实质就是每在最小生成树集合中加入一个点就需要把这个点与集合外的点比较,不断的寻找两个集合之间最小的边)

6.循环上述步骤,指导将全部顶点加入到最小生成树集合为止。

代码(OJ_1017):

#include <iostream>
#include <cstring>
using namespace std; #define INF 0x7fffffff
#define MAX 105 int N;
int graph[MAX][MAX];
int dist[MAX];
bool visited[MAX]; void prime()
{
int sum = ; memset(visited, false, sizeof(visited)); //初始化visited
visited[] = true; for(int i = ; i <= N; i++)
{
if(visited[i] == false && dist[i] > graph[][i]) //初始化dist[i]
dist[i] = graph[][i];
} for(int i = ; i < N; i++) //找生成树集合点集相连最小权值的边
{
int min = INF, index;
for(int j = ; j <= N; j++)
{
if(visited[j] == false && min > dist[j])
{
min = dist[j];
index = j;
}
}
visited[index] = true; //加入最小生成树集合
sum += min; //记录权值之和
for(int j = ; j <= N; j++) //更新dist数组
{
if(visited[j] == false && dist[j] > graph[index][j])
dist[j] = graph[index][j];
}
} cout << sum << endl; //最小生成树权值之和
} void InitData()
{
for(int i = ; i <= N; i++)
{
for(int j = ; j <= N; j++)
{
graph[i][j] = INF;
}
dist[i] = INF;
}
} int main()
{
int a, b, value;
while(cin >> N, N)
{
InitData();
int m = (N - ) * N / ;
for(int i = ; i < m; i++)
{
cin >> a >> b >> value;
graph[a][b] = graph[b][a] = value;
}
prime();
}
}

输入:


Algorithm --> Kruskal算法和Prim算法的更多相关文章

  1. 求最小生成树——Kruskal算法和Prim算法

    给定一个带权值的无向图,要求权值之和最小的生成树,常用的算法有Kruskal算法和Prim算法.这两个算法其实都是贪心思想的使用,但又能求出最优解.(代码借鉴http://blog.csdn.net/ ...

  2. 最小生成树之Kruskal算法和Prim算法

    依据图的深度优先遍历和广度优先遍历,能够用最少的边连接全部的顶点,并且不会形成回路. 这样的连接全部顶点并且路径唯一的树型结构称为生成树或扩展树.实际中.希望产生的生成树的全部边的权值和最小,称之为最 ...

  3. 最小生成数kruskal算法和prim算法

    定义 连通图:在无向图中,若任意两个顶点vivi与vjvj都有路径相通,则称该无向图为连通图. 强连通图:在有向图中,若任意两个顶点vivi与vjvj都有路径相通,则称该有向图为强连通图. 连通网:在 ...

  4. 最小生成树的两种方法(Kruskal算法和Prim算法)

    关于图的几个概念定义: 连通图:在无向图中,若任意两个顶点vivi与vjvj都有路径相通,则称该无向图为连通图. 强连通图:在有向图中,若任意两个顶点vivi与vjvj都有路径相通,则称该有向图为强连 ...

  5. 最小生成树(次小生成树)(最小生成树不唯一) 模板:Kruskal算法和 Prim算法

    Kruskal模板:按照边权排序,开始从最小边生成树 #include<algorithm> #include<stdio.h> #include<string.h> ...

  6. 贪心算法-最小生成树Kruskal算法和Prim算法

    Kruskal算法: 不断地选择未被选中的边中权重最轻且不会形成环的一条. 简单的理解: 不停地循环,每一次都寻找两个顶点,这两个顶点不在同一个真子集里,且边上的权值最小. 把找到的这两个顶点联合起来 ...

  7. Prim算法和Dijkstra算法的异同

    Prim算法和Dijkstra算法的异同 之前一直觉得Prim和Dijkstra很相似,但是没有仔细对比: 今天看了下,主要有以下几点: 1: Prim是计算最小生成树的算法,比如为N个村庄修路,怎么 ...

  8. hdu 3864 D_num Pollard_rho算法和Miller_Rabin算法

    D_num Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Problem De ...

  9. 最短路径——Dijkstra算法和Floyd算法

    Dijkstra算法概述 Dijkstra算法是由荷兰计算机科学家狄克斯特拉(Dijkstra)于1959 年提出的,因此又叫狄克斯特拉算法.是从一个顶点到其余各顶点的最短路径算法,解决的是有向图(无 ...

随机推荐

  1. spring-java项目中连接redis数据库

    最近由于项目需要,要从redis数据库中查询一些数据,还没有了解过redis,只好硬着头皮上阵,记录一下连接过程: 1.需要导入两个jar包:jedis.jar,spring-data-redis.j ...

  2. Oracle 存储过程中的 =>

    oracle实参与形参有两种对应方式1.一种是位置方式,和面向对象语言参数传递类似;2.另外一种是=> 作为形参对应,因为位置对应方法有缺限,比如一个函数有3个参数,但第2个是可以不传(有默认值 ...

  3. JavaScript遍历table

    JavaScript遍历table 1.说明      遍历表格中的某行某列,并打印其值 2.实现源码 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML ...

  4. Linux查看网络的联机状态

    Linux查看网络的联机状态 youhaidong@youhaidong-ThinkPad-Edge-E545:~$ netstat -a^C unix 3 [ ] 流 已连接 14923 @/tmp ...

  5. 2.3 InnoDB 体系架构

    下图简单显示了InnoDB的存储引擎的体系架构,从图可见,InnoDB储存引擎有多个内存块,可以认为这些内存块组成了一个大的内存池,负责如下工作: 维护所有进程/线程需要访问的多个内部数据结构 缓存磁 ...

  6. 异常-----springmvc + ajaxfileupload解决ajax不能异步上传图片的问题。java.lang.ClassCastException: org.apache.catalina.connector.RequestFacade cannot be cast to org.springframework.web.multipart.

    说明这个问题产生的原因主要是form表单上传图片的时候必须是Content-Type:"multipart/form-data,这种格式的,但是ajax在页面不刷新的情况下去加载的时候只会把 ...

  7. [APIO2009]会议中心

    [APIO2009]会议中心 题目大意: 原网址与样例戳我! 给定n个区间,询问以下问题: 1.最多能够选择多少个不相交的区间? 2.在第一问的基础上,输出字典序最小的方案. 数据范围:\(n \le ...

  8. AC自动机模板1(【洛谷3808】)

    题面 题目背景 这是一道简单的AC自动机模版题. 用于检测正确性以及算法常数. 为了防止卡OJ,在保证正确的基础上只有两组数据,请不要恶意提交. 题目描述 给定n个模式串和1个文本串,求有多少个模式串 ...

  9. NOIWC2018滚粗记

    Day0 \(THUWC\)刚刚结束..顺利的滚粗了 Day1 整理一下自己咸鱼的心 下午到学校坐车出发 这次队伍浩大THUWC只有4个 又到了雅礼洋湖这个地方 路上突然多了许多有关\(NOI\)的标 ...

  10. [BZOJ1602] [Usaco2008 Oct] 牧场行走 (LCA)

    Description N头牛(2<=n<=1000)别人被标记为1到n,在同样被标记1到n的n块土地上吃草,第i头牛在第i块牧场吃草. 这n块土地被n-1条边连接. 奶牛可以在边上行走, ...