Algorithm --> Kruskal算法和Prim算法
最小生成树之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算法的更多相关文章
- 求最小生成树——Kruskal算法和Prim算法
给定一个带权值的无向图,要求权值之和最小的生成树,常用的算法有Kruskal算法和Prim算法.这两个算法其实都是贪心思想的使用,但又能求出最优解.(代码借鉴http://blog.csdn.net/ ...
- 最小生成树之Kruskal算法和Prim算法
依据图的深度优先遍历和广度优先遍历,能够用最少的边连接全部的顶点,并且不会形成回路. 这样的连接全部顶点并且路径唯一的树型结构称为生成树或扩展树.实际中.希望产生的生成树的全部边的权值和最小,称之为最 ...
- 最小生成数kruskal算法和prim算法
定义 连通图:在无向图中,若任意两个顶点vivi与vjvj都有路径相通,则称该无向图为连通图. 强连通图:在有向图中,若任意两个顶点vivi与vjvj都有路径相通,则称该有向图为强连通图. 连通网:在 ...
- 最小生成树的两种方法(Kruskal算法和Prim算法)
关于图的几个概念定义: 连通图:在无向图中,若任意两个顶点vivi与vjvj都有路径相通,则称该无向图为连通图. 强连通图:在有向图中,若任意两个顶点vivi与vjvj都有路径相通,则称该有向图为强连 ...
- 最小生成树(次小生成树)(最小生成树不唯一) 模板:Kruskal算法和 Prim算法
Kruskal模板:按照边权排序,开始从最小边生成树 #include<algorithm> #include<stdio.h> #include<string.h> ...
- 贪心算法-最小生成树Kruskal算法和Prim算法
Kruskal算法: 不断地选择未被选中的边中权重最轻且不会形成环的一条. 简单的理解: 不停地循环,每一次都寻找两个顶点,这两个顶点不在同一个真子集里,且边上的权值最小. 把找到的这两个顶点联合起来 ...
- Prim算法和Dijkstra算法的异同
Prim算法和Dijkstra算法的异同 之前一直觉得Prim和Dijkstra很相似,但是没有仔细对比: 今天看了下,主要有以下几点: 1: Prim是计算最小生成树的算法,比如为N个村庄修路,怎么 ...
- 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 ...
- 最短路径——Dijkstra算法和Floyd算法
Dijkstra算法概述 Dijkstra算法是由荷兰计算机科学家狄克斯特拉(Dijkstra)于1959 年提出的,因此又叫狄克斯特拉算法.是从一个顶点到其余各顶点的最短路径算法,解决的是有向图(无 ...
随机推荐
- 【转载】SDL2.0在mfc窗口中显示yuv的一种方法
DWORD ThreadFun(){ //用mfc窗口句柄创建一个sdl window SDL_Window * pWindow = SDL_CreateWindowFrom( (void ...
- 错误代码: 1052 Column 'stu_id' in field list is ambiguous
1.错误描述 1 queries executed, 0 success, 1 errors, 0 warnings 查询:select stu_id, (SELECT stu_name FROM t ...
- eclipse和android studio的爱恨情仇
Eclipse,以下简称ES(自己起的,不喜勿喷):Android studio,以下简称AS(都这么叫的啦)! 2000年,IBM怀胎24个月,终于产生了Eclipse,当时ES的诞生只是为了解决I ...
- ASP.NET CORE的Code Fist后Models更改了怎么办?
上次我写到MVC的code fist后,自动生成数据库并自动生成web页面了 点击打开链接 那么随着项目需求的逐步明确,model变化了怎么办呢?其实和上次一样的,有两条关键的语句要记住 Add-Mi ...
- 关于CS1061报错(XX不包含XXX的定义,并且找不到类型为XX的第一个参.....)的一种可能的解决的办法
在我编程中,我遇到了一个这样的报错, 可是我引用的product类中又确实定义了这么一个方法, protected void BindPageData(int categoryID) { Produc ...
- tomcat原理(一)server.xml中的host虚拟主机的理解
一.Tomcat服务器端口的配置 Tomcat的所有配置都放在conf文件夹之中,里面的server.xml文件是配置的核心文件. 如果想修改Tomcat服务器的启动端口,则可以在server.xml ...
- 关于access_token过期的解决办法
最近在做微信的发送模版消息,在测试的时候发现有的时候能够发送,有时候无法发送,查了相关的日志(日志记录发送结果很重要!!),看到了微信返回的错误消息,发现是 invalid credential, a ...
- 【Luogu1879】玉米田(状态压缩,动态规划)
懒得搞题目了 哦对了,这题双倍经验 题解 装压DP 利用位运算很容易解决相邻位的问题 其实我的还是太复杂了 具体的,更加好的位运算的写法可以参考YL大佬,但是我也搞不到他代码,因为他太强了. 然而他博 ...
- [BZOJ1003] [ZJOI2006] 物流运输trans (最短路 & dp)
Description 物流公司要把一批货物从码头A运到码头B.由于货物量比较大,需要n天才能运完.货物运输过程中一般要转停好几个码头.物流公司通常会设计一条固定的运输路线,以便对整个运输过程实施严格 ...
- 剑指offer(15)反转链表
题目描述 输入一个链表,反转链表后,输出链表的所有元素. 题目分析 至少需要三个指针pPre(指向前一个结点).pCurrent(指向当前的结点,在代码中就是pHead).pPnext(指向后一个结点 ...