最小生成树

通俗解释:一个连通图,可将这个连通图删减任意条边,仍然保持连通图的状态并且所有边权值加起来的总和使其达到最小。这就是最小生成树

可以参考下图,便于理解

原来的图:



最小生成树(蓝色线):

最小生成树主要有prim和kruskal两种算法

其中prim可以用优先队列实现,kruskal使用并查集来实现

两种算法针对于不同的数据规模有不同的效率,根据不同的题目可以选择相应的算法。

经典最小生成树算法应用的案例如HDU-1863这个问题

概述

省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可)。经过调查评估,得到的统计表中列出了有可能建设公路的若干条道路的成本。现请你编写程序,计算出全省畅通需要的最低成本。

输入

测试输入包含若干测试用例。每个测试用例的第1行给出评估的道路条数 N、村庄数目M ( < 100 );随后的 N

行对应村庄间道路的成本,每行给出一对正整数,分别是两个村庄的编号,以及此两村庄间道路的成本(也是正整数)。为简单起见,村庄从1到M编号。当N为0时,全部输入结束,相应的结果不要输出。

输出

对每个测试用例,在1行里输出全省畅通需要的最低成本。若统计数据不足以保证畅通,则输出“?”。

输入样例

3 3

1 2 1

1 3 2

2 3 4

1 3

2 3 2

0 100

输出样例

3

?

题目来源于杭电HDU HDU-1863

这道题其实就是要求计算出最小生成树的所有路径长度

可以直接在计算过程中累加即可,接下来介绍这两种算法并试着如何解决这个问题:

prim算法

prim算法需要依赖邻接表,以及存储边的优先队列(原理基本上等同于堆排序,实际上用数组排序也可以,但考虑到时间复杂度推荐使用优先队列),总体来说代码比较容易,反复练习几遍基本上就能掌握

其中类似于广搜的思路,代码如下:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<queue>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<vector>
using namespace std;
#define pb push_back
#define mp make_pair //定义结构体用来存储边
struct nod{
int x,y;
int val;
nod(){};
nod(int x,int y,int v):x(x),y(y),val(v){};
};
//使优先队列从小到大排序,这里注意应该使用大于号,因为优先队列是默认大的先出队
bool operator<(nod a,nod b){
return a.val>b.val;
} int co;
int cou;
int guess; //定义类型
typedef vector<pair<int,int> > ve_pa; ve_pa ve[1000]; //邻接表
priority_queue<nod> pq; //优先队列给已加入的边进行排序
int have[1000]; //该点是否已经计算完成 int main(){
int line;
while(cin>>line>>co){
if(line==0)break;
//初始化
while(!pq.empty())pq.pop();
for(int i=1;i<=co;i++)ve[i].clear();
memset(have,0,sizeof(have));
cou=0;
guess=1; //获取输入,存入邻接表
for(int i=1;i<=line;i++){
int a,b,c;
scanf("%d%d%d",&a,&b,&c);
ve[a].pb(mp(b,c));
ve[b].pb(mp(a,c));
} //起始队列插入
have[1]=1;
for(ve_pa::iterator vi=ve[1].begin();vi!=ve[1].end();vi++){
pq.push(nod(1,vi->first,vi->second));
} //开始计算队列
while(!pq.empty()){
if(guess>=co)break; //点是否全部计算完成
nod temp=pq.top();
pq.pop();
if(have[temp.y])continue; //该点是否计算完成
have[temp.y]=1; //该点已计算完成
guess++; //已计算的点数+1
cou+=temp.val; //记录最小生成树的权值总和
for(ve_pa::iterator vi=ve[temp.y].begin();vi!=ve[temp.y].end();vi++){
//从邻接表对该点进行遍历,加入队列
pq.push(nod(temp.y,vi->first,vi->second));
}
} //如果非连通图,则输出?
if(guess!=co){
cout<<"?"<<endl;
}else{
cout<<cou<<endl;
}
}
}

kruskal算法

kruskal算法依赖并查集,用并查集来判断图是否存在回路。

该算法并不需要邻接表,仅需存储边即可。在算法中需要按照边长短做一次排序。然后依次从小到大并查集合并,每次合并时把边权值加入到统计就能求出结果。

掌握了并查集之后思路也好理解,代码如下:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<queue>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<vector>
using namespace std;
#define pb push_back
#define mp make_pair //定义结构体用于存储所有的边
struct nod{
int a,b,v;
nod(){};
nod(int a,int b,int v):a(a),b(b),v(v){};
}
//对边从小到大排序
bool operator < (nod a,nod b){
return a.v<b.v;
} int fa[1000]; //并查集中该点的父节点下标
int sum=0,times=0;
nod nods[5000]; //并查集两个函数
int uf_find(int a){
if(a==fa[a])return a;
return fa[a]=uf_find(fa[a]);
}
int uf_union(int a,int b){
int ra=uf_find(a),rb=uf_find(b);
if(ra!=rb){
fa[rb]=ra;
}
} int main(){
int co,line;
while(cin>>line>>co){
if(line==0)break;
//初始化
for(int i=1;i<=co;i++)fa[i]=i;
sum=0;
times=1;
//读取输入
for(int i=0;i<line;i++){
int a,b,c;
scanf("%d%d%d",&nods[i].a,&nods[i].b,&nods[i].v);
}
//排序
sort(nods,nods+line);
//从小到大遍历
for(int i=0;i<line;i++){
if(times>=co)break;//剪枝
//如果该边对应的两个点是一个集合里的,则跳过
if(uf_find(nods[i].a)==uf_find(nods[i].b))continue;
//合并
uf_union(nods[i].a,nods[i].b);
//记录最小生成树的权值总和
sum+=nods[i].v;
times++;
} //如果非连通图,则输出?
if(times<co){
cout<<"?"<<endl;
}else{
cout<<sum<<endl;
}
}
}

最小生成树算法 prim kruskal两种算法实现 HDU-1863 畅通工程的更多相关文章

  1. hdu 1863 畅通工程 (并查集+最小生成树)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1863 畅通工程 Time Limit: 1000/1000 MS (Java/Others)    M ...

  2. HDU 1863 畅通工程(Prim算法求解MST)

    题目: 省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可).经过调查评估,得到的统计表中列出了有可能建设公路的若干条道路的成本.现 ...

  3. HDU 1863 畅通工程 克鲁斯卡尔算法

    畅通工程 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  4. HDU 1863 畅通工程(Prim,Kruskal,邻接表模板)

    畅通工程 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  5. HDU 1863 畅通工程 最小生成树

    思路: 比较典型的最小生成树的题目了..在这里用求最小生成树的经典算法K(Kruskal)算法和P(Prim)算法.我的 K 算法用的是结构体来存图,P 算法用的是邻接矩阵来存图,K算法的复杂度是O( ...

  6. HDU - 1863 畅通工程(最小生成树)

    d.m个村庄,n条路,计算出所有村庄畅通需要的最低成本. s.最小生成树 c.Prim算法:cost[a][b]和cost[b][a]都得赋值. /* Prim算法 Prim求MST 耗费矩阵cost ...

  7. HDU 1863 畅通工程 -Kruskal模版

    畅通工程 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  8. hdu 1863 畅通工程(Kruskal+并查集)

    畅通工程 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submi ...

  9. hdu 1863 畅通工程 (并查集 、 kruskal)

    畅通工程Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...

随机推荐

  1. 数据库:sql语句分别按日,按周,按月,按季统计金额

    如: 表:consume_record 字段:consume (money类型) date (datetime类型) 请问怎么写四条sql语句分别按日,按周,按月,按季统计消费总量. 如:1月 120 ...

  2. MySQL组合索引最左匹配原则

    几个重要的概念 1.对于mysql来说,一条sql中,一个表无论其蕴含的索引有多少,但是有且只用一条. 2.对于多列索引来说(a,b,c)其相当于3个索引(a),(a,b),(a,b,c)3个索引,又 ...

  3. WPF TextBox 一些设置技巧

    WPF TextBox 一些设置技巧 运行环境:Win10 x64, NetFrameWork 4.8, 作者:乌龙哈里,日期:2019-05-01 参考: 章节: 取消输入法 输入方式设定为Over ...

  4. Windows版本Apache+php的Xhprof应用__[2]

    [计划] “Windows版本Apache+php的Xhprof应用__[1]”中已经解决了下载,配置的问题,所以这里的工作是接着进行的,我们以调试一个 php代码的文件来看看是怎么用xhprof的. ...

  5. 【转】nginx的模块变量(HTTP核心模块变量)

    nginx的HTTP核心模块引入了大量的变量,可以在指定范围内使用这些变量的值,可以分为三类:一是客户请求头中发送的变量.二是服务器端响应头中的变量,第三是nginx产生的各种变量,我们可以使用$变量 ...

  6. redis 有用

     浅谈redis   (1)什么是redis? Redis 是一个基于内存的高性能key-value数据库. (有空再补充,有理解错误或不足欢迎指正)   (2)Reids的特点 redis本质上是一 ...

  7. Linux tput命令

    一.简介 shell 脚本编写者往往需要能通过一种方法将输出更改为粗体,为其加下划线,实现反向突出显示等,这正是 tput 的用武之地. tput 命令将通过 terminfo 数据库对您的终端会话进 ...

  8. Luogu 4552 [Poetize6] IncDec Sequence

    在BZOJ上好像被权限掉了. 考虑差分,定义差分数组$b$ $$b_i = \left\{\begin{matrix} a_i \ \ \ (i == 1)\\ a_i - a_{i - 1}\ \ ...

  9. scala中的表达式

    scala中的表达式是有值的, 所以可以把表达式当做参数来传递, 那么接受表达式的形参定义一般是: block: =>Unit   , 没有形参,返回类型Unit spark中的这个代码很经典, ...

  10. review backpropagation

    The goal of backpropagation is to compute the partial derivatives ∂C/∂w and ∂C/∂b of the cost functi ...