灾后重建

题目描述

Pear市一共有N(<=50000)个居民点,居民点之间有M(<=200000)条双向道路相连。这些居民点两两之间都可以通过双向道路到达。这种情况一直持续到最近,一次严重的地震毁坏了全部M条道路。

震后,Pear打算修复其中一些道路,修理第i条道路需要Pi的时间。不过,Pear并不打算让全部的点连通,而是选择一些标号特殊的点让他们连通。

Pear有Q(<=50000)次询问,每次询问,他会选择所有编号在[l,r]之间,并且 编号 mod K = C 的点,修理一些路使得它们连通。由于所有道路的修理可以同时开工,所以完成修理的时间取决于花费时间最长的一条路,即涉及到的道路中Pi的最大值。

你能帮助Pear计算出每次询问时需要花费的最少时间么?这里询问是独立的,也就是上一个询问里的修理计划并没有付诸行动。

【输入格式】

第一行三个正整数N、M、Q,含义如题面所述。

接下来M行,每行三个正整数Xi、Yi、Pi,表示一条连接Xi和Yi的双向道路,修复需要Pi的时间。可能有自环,可能有重边。1<=Pi<=1000000。

接下来Q行,每行四个正整数Li、Ri、Ki、Ci,表示这次询问的点是[Li,Ri]区间中所有编号Mod Ki=Ci的点。保证参与询问的点至少有两个。

【输出格式】

输出Q行,每行一个正整数表示对应询问的答案。

【样例输入】

7 10 4

1 3 10

2 6 9

4 1 5

3 7 4

3 6 9

1 5 8

2 7 4

3 2 10

1 7 6

7 6 9

1 7 1 0

1 7 3 1

2 5 1 0

3 7 2 1

【样例输出】

9

6

8

8

【数据范围】

对于20%的数据,N,M,Q<=30

对于40%的数据,N,M,Q<=2000

对于100%的数据,N<=50000,M<=2*10^5,Q<=50000. Pi<=10^6. Li,Ri,Ki均在[1,N]范围内,Ci在[0,对应询问的Ki)范围内。

资源约定:

峰值内存消耗(含虚拟机) < 256M

CPU消耗 < 5000ms

请严格按要求输出,不要画蛇添足地打印类似:“请您输入…” 的多余内容。

所有代码放在同一个源文件中,调试通过后,拷贝提交该源码。

注意:不要使用package语句。不要使用jdk1.7及以上版本的特性。

注意:主类的名字必须是:Main,否则按无效代码处理。

  1. import java.util.ArrayList;
  2. import java.util.Scanner;
  3. public class Main {
  4. //使用Prim算法,获取输入图的最小生成树
  5. public int[][] getPrim(int[][] value) {
  6. int[][] result = new int[value.length][value[0].length]; //存放最终最小生成树的边权值
  7. int[] used = new int[value.length]; //用于判断顶点是否被遍历
  8. for(int i = 1, len = value.length;i < len;i++)
  9. used[i] = -1; //初始化,所有顶点均未被遍历
  10. used[1] = 1; //从顶点1开始遍历,表示顶点已经被遍历
  11. int count = 1; //记录已经完成构造最小生成树的顶点
  12. int len = value.length;
  13. while(count < len) { //当已经遍历的顶点个数达到图的顶点个数len时,退出循环
  14. int tempMax = Integer.MAX_VALUE;
  15. int tempi = 0;
  16. int tempj = 0;
  17. for(int i = 1;i < len;i++) { //用于遍历已经构造的顶点
  18. if(used[i] == -1)
  19. continue;
  20. for(int j = 1;j < len;j++) { //用于遍历未构造的顶点
  21. if(used[j] == -1) {
  22. if(value[i][j] != 0 && tempMax > value[i][j]) {
  23. tempMax = value[i][j];
  24. tempi = i;
  25. tempj = j;
  26. }
  27. }
  28. }
  29. }
  30. result[tempi][tempj] = tempMax;
  31. result[tempj][tempi] = tempMax;
  32. used[tempj] = 1;
  33. count++;
  34. }
  35. return result;
  36. }
  37. //使用floyd算法获取所有顶点之间的最短路径的具体路径
  38. public void floyd(int[][] primTree, int[][] path) {
  39. int[][] tree = new int[primTree.length][primTree.length];
  40. for(int i = 1;i < primTree.length;i++)
  41. for(int j = 1;j < primTree.length;j++)
  42. tree[i][j] = primTree[i][j];
  43. for(int k = 1;k < primTree.length;k++) {
  44. for(int i = 1;i < primTree.length;i++) {
  45. for(int j = 1;j < primTree[0].length;j++) {
  46. if(tree[i][k] != 0 && tree[k][j] != 0) {
  47. int temp = tree[i][k] + tree[k][j];
  48. if(tree[i][j] == 0) {
  49. tree[i][j] = temp;
  50. path[i][j] = k; //存放顶点i到顶点j之间的路径节点
  51. }
  52. }
  53. }
  54. }
  55. }
  56. }
  57. //返回a与b之间的最大值
  58. public int max(int a, int b) {
  59. return a > b ? a : b;
  60. }
  61. //根据最短路径,返回顶点start~end之间的最大权值边
  62. public int dfsMax(int[][] primTree, int[][] path, int start, int end) {
  63. if(path[start][end] == 0)
  64. return primTree[start][end];
  65. int mid = path[start][end]; //start和end的中间顶点
  66. return max(dfsMax(primTree, path, start, mid), dfsMax(primTree, path, mid, end));
  67. }
  68. //根据最小生成树,返回各个顶点到其它顶点行走过程中,权值最大的一条边
  69. public int[][] getMaxValue(int[][] primTree) {
  70. int[][] path = new int[primTree.length][primTree[0].length];
  71. floyd(primTree, path); //获取具体最短路径
  72. int[][] result = new int[primTree.length][primTree[0].length];
  73. for(int i = 1;i < primTree.length;i++) {
  74. for(int j = 1;j < primTree.length;j++) {
  75. if(j == i)
  76. continue;
  77. int max = dfsMax(primTree, path, i, j);
  78. result[i][j] = max;
  79. }
  80. }
  81. return result;
  82. }
  83. //打印出题意结果
  84. public void printResult(int[][] value, int[][] result) {
  85. int[][] primTree = getPrim(value); //获取输入图的最小生成树
  86. int[][] maxResult = getMaxValue(primTree); //获取各个顶点到其它顶点最短路径中最大权值边
  87. for(int i = 0;i < result.length;i++) {
  88. int L = result[i][0];
  89. int R = result[i][1];
  90. int K = result[i][2];
  91. int C = result[i][3];
  92. ArrayList<Integer> list = new ArrayList<Integer>();
  93. for(int j = L;j <= R;j++) {
  94. if(j % K == C)
  95. list.add(j);
  96. }
  97. int max = 0;
  98. for(int j = 0;j < list.size();j++) {
  99. for(int k = j + 1;k < list.size();k++) {
  100. if(max < maxResult[list.get(j)][list.get(k)])
  101. max = maxResult[list.get(j)][list.get(k)];
  102. }
  103. }
  104. System.out.println(max);
  105. }
  106. return;
  107. }
  108. public static void main(String[] args) {
  109. Main test = new Main();
  110. Scanner in = new Scanner(System.in);
  111. int N = in.nextInt();
  112. int M = in.nextInt();
  113. int Q = in.nextInt();
  114. int[][] value = new int[N + 1][N + 1];
  115. for(int i = 1;i <= M;i++) {
  116. int a = in.nextInt();
  117. int b = in.nextInt();
  118. int tempV = in.nextInt();
  119. value[a][b] = tempV;
  120. value[b][a] = tempV;
  121. }
  122. int[][] result = new int[Q][4];
  123. for(int i = 0;i < Q;i++) {
  124. result[i][0] = in.nextInt();
  125. result[i][1] = in.nextInt();
  126. result[i][2] = in.nextInt();
  127. result[i][3] = in.nextInt();
  128. }
  129. test.printResult(value, result);
  130. }
  131. }

java实现第六届蓝桥杯灾后重建的更多相关文章

  1. java实现第六届蓝桥杯居民集会

    居民集会 蓝桥村的居民都生活在一条公路的边上,公路的长度为L,每户家庭的位置都用这户家庭到公路的起点的距离来计算,第i户家庭距起点的距离为di. 每年,蓝桥村都要举行一次集会.今年,由于村里的人口太多 ...

  2. java实现第六届蓝桥杯切开字符串

    切开字符串 Pear有一个字符串,不过他希望把它切成两段. 这是一个长度为N(<=10^5)的字符串. Pear希望选择一个位置,把字符串不重复不遗漏地切成两段,长度分别是t和N-t(这两段都必 ...

  3. java实现第六届蓝桥杯四阶幻方

    四阶幻方 把1~16的数字填入4x4的方格中,使得行.列以 及两个对角线的和都相等,满足这样的特征时称 为:四阶幻方. 四阶幻方可能有很多方案.如果固定左上角为1 ,请计算一共有多少种方案. 比如: ...

  4. java实现第六届蓝桥杯表格计算

    表格计算 某次无聊中, atm 发现了一个很老的程序.这个程序的功能类似于 Excel ,它对一个表格进行操作. 不妨设表格有 n 行,每行有 m 个格子. 每个格子的内容可以是一个正整数,也可以是一 ...

  5. java实现第六届蓝桥杯穿越雷区

    穿越雷区 题目描述 X星的坦克战车很奇怪,它必须交替地穿越正能量辐射区和负能量辐射区才能保持正常运转,否则将报废. 某坦克需要从A区到B区去(A,B区本身是安全区,没有正能量或负能量特征),怎样走才能 ...

  6. java实现第六届蓝桥杯密文搜索

    密文搜索 福尔摩斯从X星收到一份资料,全部是小写字母组成. 他的助手提供了另一份资料:许多长度为8的密码列表. 福尔摩斯发现,这些密码是被打乱后隐藏在先前那份资料中的. 请你编写一个程序,从第一份资料 ...

  7. java实现第六届蓝桥杯奇怪的数列

    奇怪的数列 从X星截获一份电码,是一些数字,如下: 13 1113 3113 132113 1113122113 - YY博士经彻夜研究,发现了规律: 第一行的数字随便是什么,以后每一行都是对上一行& ...

  8. java实现第六届蓝桥杯奇妙的数字

    奇妙的数字 奇妙的数字 小明发现了一个奇妙的数字.它的平方和立方正好把0~9的10个数字每个用且只用了一次. 你能猜出这个数字是多少吗? 请填写该数字,不要填写任何多余的内容. 结果:69 impor ...

  9. java实现第六届蓝桥杯循环节长度

    循环节长度 两个整数做除法,有时会产生循环小数,其循环部分称为:循环节. 比如,11/13=6=>0.846153846153..... 其循环节为[846153] 共有6位. 下面的方法,可以 ...

随机推荐

  1. 不同版本(2.3/2.4/2.5/3.0/3.1)web.xml头信息

    Web App 3.1 <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee" xmlns:xsi="http:// ...

  2. C/C++数组和指针详解

    /****************************************************************/ /*            学习是合作和分享式的! /* Auth ...

  3. 【图机器学习】cs224w Lecture 8 & 9 - 图神经网络 及 深度生成模型

    目录 Graph Neural Network Graph Convolutional Network GraphSAGE Graph Attention Network Tips Deep Gene ...

  4. Zabbix-部署

    目录 一. apt安装 Zabbix 部署结构图和主机环境 1.1 Zabbix-server 安装配置 1.1.1 安装zabbix仓库 1.1.2 安装Zabbix server.web前端.ag ...

  5. nginx配置之代理功能

    nginx代理功能 至少需要两台主机: 代理机配置----如下: yum安装的/etc/nginx/nginx.conf location / { #服务机的ip端口 proxy_pass http: ...

  6. js时间戳转为日期格式的方法

    Date.prototype.Format = function(fmt){ var o = { "M+" : this.getMonth()+1, //月份 "d+&q ...

  7. python 串口 透传

    python正常情况通过串口 serial  传输数据的时候,都是以字符串的形式发送的 str = ‘abcd’ ser.write(str.encode())#直接发送str报错,需要发送byte类 ...

  8. Pyqt5_QlineEdit

    QlineEdit 方法 setAlignment() 按固定值方式对齐文本 Qt.AlignLeft:水平方向靠左对齐 Qt.AlignRight:水平方向靠右对齐 Qt.AlignCenter:水 ...

  9. Docker 入门:容器

    容器看着像机器,实际是进程,是一个运行时程序. 要操作一个 Docker 容器,只需要执行 docker container 命令. 可以通过 help 查看 run 运行容器 基础使用: docke ...

  10. 0512String类

    String类 2. String类 2.1 字符串类型概述 又爱又恨!!! 爱: 字符串基本上就是数据的保存,传输,处理非常重要的一种手段. 恨: 解析过程非常烦人,需要掌握熟记很多方法,同时需要有 ...