灾后重建

题目描述

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实现第六届蓝桥杯灾后重建的更多相关文章

  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. AndroidStudio3.6升级后的坑-apk打包

    前段时间尝试了最新版的AndroidStudio3.6,整体来说gradle调试和自带的虚拟机相比较历史版本有了更香的体验. 刚好有个新项目,就直接使用最新版了,这次新版的升级除了保持原有的界面风格, ...

  2. c++离散化处理大范围和重复数据

    关于离散化 有些新手可能会问:离散化是什么?离散化就是将无限空间中有限的个体映射到有限的空间里去. 上面的定义肯定会有人看不懂(其实我刚开始学的时候也看不懂) 用我自己的话来说,就是在不改变数据的相对 ...

  3. Spring处理@Configuration的分析

    Spring处理@Configuration的分析 声明:本文若有任何纰漏.错误,还请不吝指出! 序言 @Configuration注解在SpringBoot中作用很大,且不说SpringBoot中的 ...

  4. 复变函数-MINDMAPS-continuous updating

  5. nginx配置之代理功能

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

  6. Django视图函数之三种响应模式

    视图函数响应处理: from django.shortcuts import render,HttpResponse,redirect (1)render 处理模板文件,可以渲染模板,第一个参数必须为 ...

  7. 安卓网络编程学习(1)——java原生网络编程(1)

    写在前面 马上要进行第二轮冲刺,考虑到自己的APP在第一轮冲刺的效果不尽人意,有很多网络方面的小BUG,这里就系统学习一下网络编程,了解来龙去脉,以便更好的对项目进行优化处理. http协议 http ...

  8. 苏浪浪 201771010120 第三周 Java基本程序设计总结

    理论知识: Java有五种语句: (1)方法调用语句(2)表达式语句(3)复合语句(4)控制语句(5)package.import语句 3.8控制流程 3.9大数值 *如果基本的整型和浮点型数据无法达 ...

  9. 转义URL 含有中文和特殊符号

    方法1: //这个方法被废弃了 NSString *urlString = @"https://www.cnblogs.com/huaida/#/程序员"; NSString* e ...

  10. C语言基础知识(三)——指针

    指针定义 1.指针的值表示的是它所指向对象的地址,指针+1表示的是下一元素的地址,按**字节**编址,而不是下一字节的地址. 2.依照数据类型而定,short占用两字节.int占用4字节.double ...