原题地址

http://acm.hdu.edu.cn/showproblem.php?pid=1233

题意:(最小生成树裸题)有N个村庄,给出村庄两两之间的距离,要求铺设公路,使得任何两个村庄间都可以实现互通(不一定有直接的公路相连,只要能间接通过公路可达即可),计算最小的公路总长度。

解题思路

上一题《HDU 1232 畅通工程》考察的是并查集的应用,这一题考察了比较重要的最小生成树算法。
最小生成树算法有两种:Prim算法和Kruskal算法,见最小生成树算法讲解。

简单总结这两种算法:

  • Prim算法从顶点的角度出发,每次选择距离当前集合最近的节点加入,直到所有节点都加入。该算法的时间复杂度为O(n²),与图中边数无关,该算法适合于稠密图
  • Kruskal算法从边的角度出发,每次总是选择权重最小的边加入,直到加入n-1条边为止。(如果加入一条边后出现回路,skip这条边)。该算法的时间主要取决于边数,它较适合于稀疏图

AC代码

Prim算法:

由于这道题目给出了所有的边数,所以可以视为稠密图。
Prim算法复杂度O(n^2),耗时296ms。

PS:Prim算法可以进行堆优化,详见Prim算法 和 堆优化的Prim算法。

#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
const int maxn = 105, INF = 0x3f3f3f;int n, map[maxn][maxn], dist[maxn]; //map[i][j]记录两顶点间距离,dist[j]记录j到V中顶点的最小值
bool visited[maxn]; //顶点是否访问过(加入了连通集)void init() //初始化距离和访问情况
{for (int i = 1; i<=n; ++i)dist[i] = INF;memset(visited, false, sizeof(visited));
}int prim() //Prim算法返回最小生成树权值之和
{int sum = 0;//以1为起点for (int i = 2; i<=n; ++i) //更新1到所有顶点的距离dist[i] = map[1][i];visited[1] = true;for (int road = 0; road < n-1; ++road) //最终会生成n-1条道路{int minn = INF, pos;for (int j = 1; j<=n; ++j) //寻找到连通集的最小边{if (!visited[j] && dist[j] < minn){minn = dist[j];pos = j; //pos记录该顶点}}visited[pos] = true;sum += dist[pos]; //加入该顶点for (int j = 1; j<=n; ++j) //松弛{if (!visited[j] && map[pos][j] < dist[j]) //更新连通集外顶点的最小距离dist[j] = map[pos][j];}}return sum;
}int main()
{ios::sync_with_stdio(false);while (cin >> n && n){init(); //初始化int u, v, tmp, ans;for (int i = 0; i < n*(n-1)/2; ++i){cin >> u >> v >> tmp;map[u][v] = map[v][u] = tmp; //顶点间距离赋值}ans = prim();cout << ans << endl;}return 0;
}

Kruskal算法(贪心+并查集)

Kruskal算法复杂度O(E*logE),E为边数。
耗时:280ms(非递归并查集),327ms(递归并查集)

#include <iostream>
#include <algorithm>
using namespace std;
const int maxn = 105;typedef struct node
{int pre, suc, w;bool operator < (const node & A) const{return w < A.w;}
}Edge; //每条边的起始、终止和权重Edge edge[maxn*maxn/2]; //记录每条边
int pre[maxn] = {0}; //记录每个顶点的上级void make_set(int n) //初始化并查集
{for (int i = 1; i <= n; ++i)pre[i] = i;
}/*
int find(int x) //递归找x的根节点
{if(pre[x] == x)return x;return pre[x] = find(pre[x]); //包含路径压缩
}
*/int find_root(int x) //非递归找x的根节点
{int r = x;while (pre[r] != r)r = pre[r];//路径压缩int i = x, j;while (i != r){j = pre[i];pre[i] = r;i = j;}return r;
}int kruskal(int n, int num) //n顶点数, num边数
{sort(edge, edge+num); //按权重升序,注意是边数make_set(n); //并查集初始化int fx, fy;int sum = 0;for (int i = 0; i < num; ++i){fx = find_root(edge[i].pre);fy = find_root(edge[i].suc);if (fx != fy) //如果不连通{pre[fy] = fx;sum += edge[i].w; //将这条边加入}}return sum;
}int main()
{ios::sync_with_stdio(false);int n, num, ans;while(cin >> n && n){num = (n * (n-1)) / 2;for (int i = 0; i < num; ++i)cin >> edge[i].pre >> edge[i].suc >> edge[i].w;ans = kruskal(n, num);cout << ans << endl;}return 0;
}

HDU 1233 还是畅通工程(最小生成树 Prim+Kruskal)相关推荐

  1. hdu 1233 还是畅通工程 最小生成树(prim算法 + kruskal算法)

    还是畅通工程                                                                            Time Limit: 4000/2 ...

  2. hdu 1233 还是畅通工程(最小生成树的Prim和Kruskal两种算法的c++实现)(prim算法详解)...

    赤裸裸滴最小生成树(MST),刚学的玩意,用两种方法熟练一下.(都是greedy) Kruskal方法:先对边按照代价非递减排序,再不断添加边且不产生环路,当边数=点数-1结束.判断加入(v,w)是否 ...

  3. HDU 1233 还是畅通工程(最小生成树)

    传送门 还是畅通工程 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total ...

  4. HDU 1863畅通工程(最小生成树)(prim算法)

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

  5. hdu 1879 继续畅通工程 最小生成树

    继续畅通工程                                                                             Time Limit: 2000/ ...

  6. HDU 1879 继续畅通工程 最小生成树

    继续畅通工程 Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Description ...

  7. hdu 1879 继续畅通工程 (最小生成树)

    继续畅通工程 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Sub ...

  8. 畅通工程//最小生成树prim

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

  9. HDU 1233 - 还是畅通工程

    Problem Description 某省调查乡村交通状况,得到的统计表中列出了任意两村庄间的距离.省政府"畅通工程"的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直 ...

最新文章

  1. 使用idea对生产环境的JAVA应用进行远程调试
  2. Visual Studio找不到iOS模拟器
  3. 8.2.1.3 Range 优化
  4. sox+linux查录音格式,linux-使用SOX和sox FAIL格式混合音频:无法打开输入文件`audio_recorded.wav’:WAVE:找不到RIFF标头...
  5. 高并发之服务降级和服务熔断____服务降级、熔断、限流的区别
  6. 惯性导航算法_自动驾驶关键技术报告:惯性导航和背后的芯片大战
  7. python PEP8规范
  8. hadoop异常 java.io.IOException: Job status not available
  9. python异步查询数据库_Python中使用aiomysql异步操作MySQL实例效果
  10. 试题16 回文数字(5位或6位)
  11. 阶段3 3.SpringMVC·_01.SpringMVC概述及入门案例_02.SpringMVC框架的介绍
  12. 硬盘分区整数计算器(如何让硬盘容量成整数)
  13. 求二元一次方程 的解 C语言
  14. errors and 0 warnings potentially fixable with the `--fix` option
  15. centos7 80端口被占用导致nginx或httpd无法启用的解决方法
  16. 如何用python画出自己喜欢的表情包
  17. WDM在不同Windows版本上的音频支持
  18. 黑莓几乎转型成纯软件公司 恢复盈利助股价暴涨16%
  19. C++详细学习笔记(二)
  20. 转-超声系统的信号链设计注意事项

热门文章

  1. 如何学习Python,CSDN帮你规划路线|Python技能树测评
  2. 常用的七个入侵检测工具推荐
  3. VR全景地图,3D实景地图能用在哪些地方?
  4. java连接SQL Server数据库失败学习小结
  5. 【考研初试的准备】复习思路 + 时间线 +专业课学习及资料搜索思路
  6. Android的ToolBar简单使用
  7. gshock手表时间校准
  8. 安卓Binder通讯
  9. 怎么用计算机唱歌少年,青少年如何学唱歌? - 备份 - 自学唱歌网
  10. 增加小球python_python实现弹跳小球