最小生成树之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. R语言︱XGBoost极端梯度上升以及forecastxgb(预测)+xgboost(回归)双案例解读

    XGBoost不仅仅可以用来做分类还可以做时间序列方面的预测,而且已经有人做的很好,可以见最后的案例. 应用一:XGBoost用来做预测 ------------------------------- ...

  2. Android WebView的缓存方式分析

    WebView的缓存可以分为(1)页面缓存和(2)数据缓存. 页面缓存是指当WebView加载一个网页时的html.JS.CSS等页面或者资源数据.这些缓存资源是由于浏览器的行为而产生,开发者只能通过 ...

  3. cmder默认的命令提示符λ改成$

    新版的cmder(2016.11.3测试)单纯修改init.bat或以前的方法都试过了不行,下面是我自己找到的方法.亲测可行. cmder\vendor\clink.lua文件中第41行中{lamb} ...

  4. Java中用正则表达式判断日期格式是否正确

    1.Java中用正则表达式判断日期格式是否正确 DateType.java: /** * @Title:DateType.java * @Package:com.you.dao * @Descript ...

  5. 采用Global.asax的Application_BeginRequest事件过滤敏感字符

    1.特殊字符过滤公共类ProcessRequest.cs using System.Web.UI; using System.Web.UI.WebControls; using System.Web. ...

  6. fineuploader使用实例

    1.Fine Uploader特点 Fine Uploader Features: A:支持文件上传进度显示. B:文件拖拽浏览器上传方式 C:Ajax页面无刷新. D:多文件上传. F:跨浏览器. ...

  7. Python Cookbook(第3版)中文版:15.20 处理C语言中的可迭代对象

    15.20 处理C语言中的可迭代对象¶ 问题¶ 你想写C扩展代码处理来自任何可迭代对象如列表.元组.文件或生成器中的元素. 解决方案¶ 下面是一个C扩展函数例子,演示了怎样处理可迭代对象中的元素: s ...

  8. javascript学习记录-2-18

    对象定义的几种方法: var  person=new Object(); person.name="111"; person.age=22; 或 var person={   na ...

  9. NOIp2017 滚粗记

    NOIp2017 滚粗记 Day0 早上 早自习的时候,班主任忽然告诉我们, 我们要参加期中考试... 这对于我们真是一个沉重的打击... 但是,管不着了 明天就死去考试了 上午 \(8:10\)到了 ...

  10. Js表单验证控件(使用方便,无需编码)-01使用说明

    演示地址:http://weishakeji.net/Utility/Verify/Index.htm    开源地址:https://github.com/weishakeji/Verify_Js ...