算法笔记_180:历届试题 国王的烦恼(Java)
目录
1 问题描述
如果两个小岛间的所有大桥都不能使用,则这两座小岛就不能直接到达了。然而,只要这两座小岛的居民能通过其他的桥或者其他的小岛互相到达,他们就会安然无事。但是,如果前一天两个小岛之间还有方法可以到达,后一天却不能到达了,居民们就会一起抗议。
现在C国的国王已经知道了每座桥能使用的天数,超过这个天数就不能使用了。现在他想知道居民们会有多少天进行抗议。
接下来m行,每行三个整数a, b, t,分别表示该座桥连接a号和b号两个小岛,能使用t天。小岛的编号从1开始递增。
1 2 2
1 3 2
2 3 1
3 4 3
第二天后1和2之间,以及1和3之间的桥不能使用,居民们会抗议。
第三天后3和4之间的桥不能使用,居民们会抗议。
对于50%的数据,1<=n<=500,1<=m<=10000;
对于100%的数据,1<=n<=10000,1<=m<=100000,1<=a, b<=n, 1<=t<=100000。
2 解决方案
说一下此题题意的理解问题:首先,我要吐槽一下,出题人的语文应该是体育老师教的吧。
原题描述:
现在C国的国王已经知道了每座桥能使用的天数,超过这个天数就不能使用了。现在他想知道居民们会有多少天进行抗议。
输出格式
输出一个整数,表示居民们会抗议的天数。
看到上面描述,外加题目所给实例输入及输出语句,且要确保每个岛屿都能两两到达
,第一印象就是使用最小生成树Kruskal算法。但是,这里要把边的权值变成负值,相应的最小就是该桥能够使用的天数最长,如此再求得最小生成树中的权值绝对值最小的边,即为居民开始进行抗议的天数。
带着这样的思考方式,提交后只得了10分,还是碰巧对的。
带着疑惑心里,看了一些网友的解答,又把题目读了一遍,提取关键点:居民们会抗议的天数,注意是天数,为此我理解为最小生成树中最大权值和最小权值差的绝对值就是答案。但是,网上相关网友的解答都不是这样计算的。
直到,我看到了文末参考资料一篇网友把题目描述修改了一部分,具体如下:
修改后:
现在C国的国王已经知道了每座桥能使用的天数,超过这个天数就不能使用了。现在他想知道居民们一共会发起多少次抗议。
输出
输出一个整数,表示居民们发起抗议的次数。
把这个天字改为次字,即求取最小生成树中权值不同的个数即为最终所求答案。为此,本题下面先求取最小生成树,然后用HashSet存放权值,最终set集合中元素个数即为最终答案。
下面代码在蓝桥系统中运行超时,具体原因应该是Java语言和C++语言编译运行的性能有关,大部分网友C++写的解答都是使用并查算法,其实Kruskal算法的本质也是并查算法,我用网友C++代码跑的是100分。
具体代码如下:
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashSet;
import java.util.Scanner; public class Main {
public static int n, m;
public static int[] id;
public static ArrayList<edge> list = new ArrayList<edge>();
public static HashSet<Integer> set = new HashSet<Integer>(); class MyComparator implements Comparator<edge> {
public int compare(edge arg0, edge arg1) {
if(arg0.t > arg1.t)
return 1;
else if(arg0.t < arg1.t)
return -1;
return 0;
}
} static class edge {
public int a;
public int b;
public int t; public edge(int a, int b, int t) {
this.a = a;
this.b = b;
this.t = t;
}
} public int find(int a) {
int root = a;
while(id[root] >= 0) {
root = id[root];
}
int i = 0, k = a;
while(k != root) {
i = id[k];
id[k] = root;
k = i;
}
return root;
} public void union(int a, int b) {
int rootA = find(a);
int rootB = find(b);
if(rootA == rootB)
return;
int num = id[rootA] + id[rootB];
if(id[rootA] < id[rootB]) {
id[rootB] = rootA;
id[rootA] = num;
} else {
id[rootA] = rootB;
id[rootB] = num;
}
} public void kruskal() {
Collections.sort(list, new MyComparator());
id = new int[n + 1];
for(int i = 1;i <= n;i++)
id[i] = -1;
int count = 0;
for(int i = 0;i < list.size();i++) {
edge p = list.get(i);
if(find(p.a) != find(p.b)) {
set.add(p.t);
union(p.a, p.b);
count++;
if(count == n - 1)
break;
}
}
System.out.println(set.size());
} public static void main(String[] args) {
Main test = new Main();
Scanner in = new Scanner(System.in);
n = in.nextInt();
m = in.nextInt();
for(int i = 0;i < m;i++) {
int a = in.nextInt();
int b = in.nextInt();
int t = in.nextInt();
list.add(new edge(a, b, -1 * t));
}
test.kruskal();
}
}
参考资料:
最新文章
- telnet模拟邮件发送
- 用PS设计等高线效果的背景图片
- R中的name命名系列函数总结
- 查询数组里有多少个数在[L,R]范围中(二分)
- zhx&#39;s contest (矩阵快速幂 + 数学推论)
- hdu 5738 2016 Multi-University Training Contest 2 Eureka 计数问题(组合数学+STL)
- inline-block元素的一些坑
- Swift - 05 - 数值型字面量
- html5 绘制集合图形
- 具体说明 Flume介绍、安装和配置
- 对synchronized关键字的理解
- Linux配置JDK
- HTML5 模拟现实物理效果
- WEB前端开发常见问题汇总
- Android Studio 将文件打包成APK
- kafka 流式计算
- python中的深拷贝和浅拷贝
- alpha冲刺(5/10)
- Mordern Effective C++ --auto
- CentOS7安装配置PostgreSQL9.6
热门文章
- ThinkPHP中I(&#39;post.&#39;)与create()方法的对比
- Tasker App Factory
- 关于错误errno EFAULT:Bad address
- centos安装sqlserver
- 解决ElasticSearch5.x中@Field注解之IK分词不能用的问题
- Android - 警告Nested weights are bad for performance
- 配置使用ldap中碰到的各种问题 --- 吐血
- 解决谷歌google不能访问造成wordpress打开缓慢的问题
- 数据库实例: STOREBOOK >; 表空间 >; 编辑 表空间: USERS
- iOS:UIToolBar、toolbarItems、BarButtonItem的几种关系