java实现第六届蓝桥杯灾后重建
灾后重建
题目描述
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,否则按无效代码处理。
import java.util.ArrayList;
import java.util.Scanner;
public class Main {
//使用Prim算法,获取输入图的最小生成树
public int[][] getPrim(int[][] value) {
int[][] result = new int[value.length][value[0].length]; //存放最终最小生成树的边权值
int[] used = new int[value.length]; //用于判断顶点是否被遍历
for(int i = 1, len = value.length;i < len;i++)
used[i] = -1; //初始化,所有顶点均未被遍历
used[1] = 1; //从顶点1开始遍历,表示顶点已经被遍历
int count = 1; //记录已经完成构造最小生成树的顶点
int len = value.length;
while(count < len) { //当已经遍历的顶点个数达到图的顶点个数len时,退出循环
int tempMax = Integer.MAX_VALUE;
int tempi = 0;
int tempj = 0;
for(int i = 1;i < len;i++) { //用于遍历已经构造的顶点
if(used[i] == -1)
continue;
for(int j = 1;j < len;j++) { //用于遍历未构造的顶点
if(used[j] == -1) {
if(value[i][j] != 0 && tempMax > value[i][j]) {
tempMax = value[i][j];
tempi = i;
tempj = j;
}
}
}
}
result[tempi][tempj] = tempMax;
result[tempj][tempi] = tempMax;
used[tempj] = 1;
count++;
}
return result;
}
//使用floyd算法获取所有顶点之间的最短路径的具体路径
public void floyd(int[][] primTree, int[][] path) {
int[][] tree = new int[primTree.length][primTree.length];
for(int i = 1;i < primTree.length;i++)
for(int j = 1;j < primTree.length;j++)
tree[i][j] = primTree[i][j];
for(int k = 1;k < primTree.length;k++) {
for(int i = 1;i < primTree.length;i++) {
for(int j = 1;j < primTree[0].length;j++) {
if(tree[i][k] != 0 && tree[k][j] != 0) {
int temp = tree[i][k] + tree[k][j];
if(tree[i][j] == 0) {
tree[i][j] = temp;
path[i][j] = k; //存放顶点i到顶点j之间的路径节点
}
}
}
}
}
}
//返回a与b之间的最大值
public int max(int a, int b) {
return a > b ? a : b;
}
//根据最短路径,返回顶点start~end之间的最大权值边
public int dfsMax(int[][] primTree, int[][] path, int start, int end) {
if(path[start][end] == 0)
return primTree[start][end];
int mid = path[start][end]; //start和end的中间顶点
return max(dfsMax(primTree, path, start, mid), dfsMax(primTree, path, mid, end));
}
//根据最小生成树,返回各个顶点到其它顶点行走过程中,权值最大的一条边
public int[][] getMaxValue(int[][] primTree) {
int[][] path = new int[primTree.length][primTree[0].length];
floyd(primTree, path); //获取具体最短路径
int[][] result = new int[primTree.length][primTree[0].length];
for(int i = 1;i < primTree.length;i++) {
for(int j = 1;j < primTree.length;j++) {
if(j == i)
continue;
int max = dfsMax(primTree, path, i, j);
result[i][j] = max;
}
}
return result;
}
//打印出题意结果
public void printResult(int[][] value, int[][] result) {
int[][] primTree = getPrim(value); //获取输入图的最小生成树
int[][] maxResult = getMaxValue(primTree); //获取各个顶点到其它顶点最短路径中最大权值边
for(int i = 0;i < result.length;i++) {
int L = result[i][0];
int R = result[i][1];
int K = result[i][2];
int C = result[i][3];
ArrayList<Integer> list = new ArrayList<Integer>();
for(int j = L;j <= R;j++) {
if(j % K == C)
list.add(j);
}
int max = 0;
for(int j = 0;j < list.size();j++) {
for(int k = j + 1;k < list.size();k++) {
if(max < maxResult[list.get(j)][list.get(k)])
max = maxResult[list.get(j)][list.get(k)];
}
}
System.out.println(max);
}
return;
}
public static void main(String[] args) {
Main test = new Main();
Scanner in = new Scanner(System.in);
int N = in.nextInt();
int M = in.nextInt();
int Q = in.nextInt();
int[][] value = new int[N + 1][N + 1];
for(int i = 1;i <= M;i++) {
int a = in.nextInt();
int b = in.nextInt();
int tempV = in.nextInt();
value[a][b] = tempV;
value[b][a] = tempV;
}
int[][] result = new int[Q][4];
for(int i = 0;i < Q;i++) {
result[i][0] = in.nextInt();
result[i][1] = in.nextInt();
result[i][2] = in.nextInt();
result[i][3] = in.nextInt();
}
test.printResult(value, result);
}
}
java实现第六届蓝桥杯灾后重建的更多相关文章
- java实现第六届蓝桥杯居民集会
居民集会 蓝桥村的居民都生活在一条公路的边上,公路的长度为L,每户家庭的位置都用这户家庭到公路的起点的距离来计算,第i户家庭距起点的距离为di. 每年,蓝桥村都要举行一次集会.今年,由于村里的人口太多 ...
- java实现第六届蓝桥杯切开字符串
切开字符串 Pear有一个字符串,不过他希望把它切成两段. 这是一个长度为N(<=10^5)的字符串. Pear希望选择一个位置,把字符串不重复不遗漏地切成两段,长度分别是t和N-t(这两段都必 ...
- java实现第六届蓝桥杯四阶幻方
四阶幻方 把1~16的数字填入4x4的方格中,使得行.列以 及两个对角线的和都相等,满足这样的特征时称 为:四阶幻方. 四阶幻方可能有很多方案.如果固定左上角为1 ,请计算一共有多少种方案. 比如: ...
- java实现第六届蓝桥杯表格计算
表格计算 某次无聊中, atm 发现了一个很老的程序.这个程序的功能类似于 Excel ,它对一个表格进行操作. 不妨设表格有 n 行,每行有 m 个格子. 每个格子的内容可以是一个正整数,也可以是一 ...
- java实现第六届蓝桥杯穿越雷区
穿越雷区 题目描述 X星的坦克战车很奇怪,它必须交替地穿越正能量辐射区和负能量辐射区才能保持正常运转,否则将报废. 某坦克需要从A区到B区去(A,B区本身是安全区,没有正能量或负能量特征),怎样走才能 ...
- java实现第六届蓝桥杯密文搜索
密文搜索 福尔摩斯从X星收到一份资料,全部是小写字母组成. 他的助手提供了另一份资料:许多长度为8的密码列表. 福尔摩斯发现,这些密码是被打乱后隐藏在先前那份资料中的. 请你编写一个程序,从第一份资料 ...
- java实现第六届蓝桥杯奇怪的数列
奇怪的数列 从X星截获一份电码,是一些数字,如下: 13 1113 3113 132113 1113122113 - YY博士经彻夜研究,发现了规律: 第一行的数字随便是什么,以后每一行都是对上一行& ...
- java实现第六届蓝桥杯奇妙的数字
奇妙的数字 奇妙的数字 小明发现了一个奇妙的数字.它的平方和立方正好把0~9的10个数字每个用且只用了一次. 你能猜出这个数字是多少吗? 请填写该数字,不要填写任何多余的内容. 结果:69 impor ...
- java实现第六届蓝桥杯循环节长度
循环节长度 两个整数做除法,有时会产生循环小数,其循环部分称为:循环节. 比如,11/13=6=>0.846153846153..... 其循环节为[846153] 共有6位. 下面的方法,可以 ...
随机推荐
- 不同版本(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:// ...
- C/C++数组和指针详解
/****************************************************************/ /* 学习是合作和分享式的! /* Auth ...
- 【图机器学习】cs224w Lecture 8 & 9 - 图神经网络 及 深度生成模型
目录 Graph Neural Network Graph Convolutional Network GraphSAGE Graph Attention Network Tips Deep Gene ...
- Zabbix-部署
目录 一. apt安装 Zabbix 部署结构图和主机环境 1.1 Zabbix-server 安装配置 1.1.1 安装zabbix仓库 1.1.2 安装Zabbix server.web前端.ag ...
- nginx配置之代理功能
nginx代理功能 至少需要两台主机: 代理机配置----如下: yum安装的/etc/nginx/nginx.conf location / { #服务机的ip端口 proxy_pass http: ...
- js时间戳转为日期格式的方法
Date.prototype.Format = function(fmt){ var o = { "M+" : this.getMonth()+1, //月份 "d+&q ...
- python 串口 透传
python正常情况通过串口 serial 传输数据的时候,都是以字符串的形式发送的 str = ‘abcd’ ser.write(str.encode())#直接发送str报错,需要发送byte类 ...
- Pyqt5_QlineEdit
QlineEdit 方法 setAlignment() 按固定值方式对齐文本 Qt.AlignLeft:水平方向靠左对齐 Qt.AlignRight:水平方向靠右对齐 Qt.AlignCenter:水 ...
- Docker 入门:容器
容器看着像机器,实际是进程,是一个运行时程序. 要操作一个 Docker 容器,只需要执行 docker container 命令. 可以通过 help 查看 run 运行容器 基础使用: docke ...
- 0512String类
String类 2. String类 2.1 字符串类型概述 又爱又恨!!! 爱: 字符串基本上就是数据的保存,传输,处理非常重要的一种手段. 恨: 解析过程非常烦人,需要掌握熟记很多方法,同时需要有 ...