HDOJ 1879 继续畅通工程 Prim算法或Kruskal算法

2023-10-17 09:32

本文主要是介绍HDOJ 1879 继续畅通工程 Prim算法或Kruskal算法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Problem Description
省政府“畅通工程”的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可)。现得到城镇道路统计表,表中列出了任意两城镇间修建道路的费用,以及该道路是否已经修通的状态。现请你编写程序,计算出全省畅通需要的最低成本。
Input
测试输入包含若干测试用例。每个测试用例的第1行给出村庄数目N ( 1< N < 100 );随后的 N(N-1)/2 行对应村庄间道路的成本及修建状态,每行给4个正整数,分别是两个村庄的编号(从1编号到N),此两村庄间道路的成本,以及修建状态:1表示已建,0表示未建。
当N为0时输入结束。
Output
每个测试用例的输出占一行,输出全省畅通需要的最低成本。
Sample Input
3
1 2 1 0
1 3 2 0
2 3 4 0
3
1 2 1 0
1 3 2 0
2 3 4 1
3
1 2 1 0
1 3 2 1
2 3 4 1
0
Sample Output
3
1
0

解题思路:
Prim算法的一点小改变,在寻找将某一结点加入已连通集合的最小代价时,如果该结点与已连通集合中的某一结点有已经修好的边(即已连通),则最小代价为0;
当然,使用Kruskal算法也是可以的,在添加边的时候可以将那些已经连通的结点用并查集合并到同一个集合中。

AC代码(Prim算法):

#include<iostream>
#include<stdio.h>
#include<algorithm>
using namespace std;
#define INF 0x3fffffff
#define maxn 101
int map[maxn][maxn];
char map1[maxn][maxn];
int d[maxn];//用于存放连通某一顶点所需要的代价
bool mark[maxn];
int n;//顶点个数
int prim(int s) {for (int i = 1; i <= n; i++) {d[i] = INF;mark[i] = false;}int newnode = s;mark[newnode] = true;d[newnode] = 0;int ans = 0;//返回值for (int i = 1; i < n; i++) {//循环n-1次for (int j = 1; j <= n; j++) {if (mark[j] || map[newnode][j] == INF) continue;if (map1[newnode][j] == '1') {//已修建d[j] = 0;continue;}if (d[j] == INF || d[j] > map[newnode][j]) {d[j] = map[newnode][j];}}int tmp = INF;for (int j = 1; j <= n; j++) {//寻找最小值加入集合if (mark[j] || d[j] == INF) continue;if (d[j] < tmp) {tmp = d[j];newnode = j;}}if (tmp == INF) return -1;//不存在最小生成树 提前剪枝mark[newnode] = true;ans += d[newnode];}return ans;
}
void init(int x) {for (int i = 1; i <= x; i++) {for (int j = 1; j <= x; j++) {map[i][j] = INF;map1[i][j] = '0';}}
}
int main() {while (scanf("%d", &n) != EOF) {if (n == 0)break;init(n);int time = n*(n - 1) / 2;int u, v, w; char type;for (int i = 1; i <= time; i++) {scanf("%d%d%d %c", &u, &v, &w, &type);if (type == map1[u][v]) {map[u][v] = min(map[u][v], w);map[v][u] = map[u][v];}else if (type=='1') {//更新map[u][v] = w;map[v][u] = map[u][v];map1[u][v] = map1[v][u] = type;}}cout << prim(n) << endl;}return 0;
}

AC代码(Kruskal算法):

#include<iostream>
#include<stdio.h>
#include<algorithm>
using namespace std;
#define INF 0x3fffffff
#define maxn 101
int n;//顶点个数
struct edge {int u, v, w;
}e[5000];
bool cmp(edge a, edge b) { return a.w < b.w; }
int tree[maxn];
int findroot(int a) {if (tree[a] == -1)return a;else {int tmp = findroot(tree[a]);tree[a] = tmp;//路径压缩return tmp;}
}
bool hebing(int a, int b) {a = findroot(a);b = findroot(b);if (a != b) {tree[a] = b;return true;}return false;
}
void init(int x) {for (int i = 1; i <= x; i++) {tree[i] = -1;}
}
int main() {while (scanf("%d", &n) != EOF) {if (n == 0)break;init(n);int time = n*(n - 1) / 2;int u, v, w; char type;int cnt = 0;//有效边的个数for (int i = 1; i <= time; i++) {scanf("%d%d%d %c", &u, &v, &w, &type);if (type == '1') {//如果已经连通,直接合并hebing(u, v);continue;}else {e[++cnt].u = u;e[cnt].v = v;e[cnt].w = w;}}sort(e + 1, e + 1 + cnt, cmp);int ans = 0;for (int i = 1; i <= cnt; i++) {if (hebing(e[i].u, e[i].v)) {ans += e[i].w;}}cout << ans << endl;}return 0;
}

这篇关于HDOJ 1879 继续畅通工程 Prim算法或Kruskal算法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/224492

相关文章

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

使用@Cacheable注解Redis时Redis宕机或其他原因连不上继续调用原方法的解决方案

《使用@Cacheable注解Redis时Redis宕机或其他原因连不上继续调用原方法的解决方案》在SpringBoot应用中,我们经常使用​​@Cacheable​​注解来缓存数据,以提高应用的性能... 目录@Cacheable注解Redis时,Redis宕机或其他原因连不上,继续调用原方法的解决方案1

使用雪花算法产生id导致前端精度缺失问题解决方案

《使用雪花算法产生id导致前端精度缺失问题解决方案》雪花算法由Twitter提出,设计目的是生成唯一的、递增的ID,下面:本文主要介绍使用雪花算法产生id导致前端精度缺失问题的解决方案,文中通过代... 目录一、问题根源二、解决方案1. 全局配置Jackson序列化规则2. 实体类必须使用Long封装类3.

MyBatisX逆向工程的实现示例

《MyBatisX逆向工程的实现示例》本文主要介绍了MyBatisX逆向工程的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录逆向工程准备好数据库、表安装MyBATisX插件项目连接数据库引入依赖pom.XML生成实体类、

Springboot实现推荐系统的协同过滤算法

《Springboot实现推荐系统的协同过滤算法》协同过滤算法是一种在推荐系统中广泛使用的算法,用于预测用户对物品(如商品、电影、音乐等)的偏好,从而实现个性化推荐,下面给大家介绍Springboot... 目录前言基本原理 算法分类 计算方法应用场景 代码实现 前言协同过滤算法(Collaborativ

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

springboot+dubbo实现时间轮算法

《springboot+dubbo实现时间轮算法》时间轮是一种高效利用线程资源进行批量化调度的算法,本文主要介绍了springboot+dubbo实现时间轮算法,文中通过示例代码介绍的非常详细,对大家... 目录前言一、参数说明二、具体实现1、HashedwheelTimer2、createWheel3、n

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

如何通过Golang的container/list实现LRU缓存算法

《如何通过Golang的container/list实现LRU缓存算法》文章介绍了Go语言中container/list包实现的双向链表,并探讨了如何使用链表实现LRU缓存,LRU缓存通过维护一个双向... 目录力扣:146. LRU 缓存主要结构 List 和 Element常用方法1. 初始化链表2.