BZOJ1050 旅行comf(kruskal)
旅行comf
Input
Output
Sample Input
Sample Output
【样例输出1】
IMPOSSIBLE
【样例输出2】
5/4
【样例输出3】
2
解题思路:
本题给出景点数量n,道路数量m,之后给出m行,每行包括三个整数,分别为道路两端的景点,x,y与该道路的行驶速度v,下一行为两个整数,分别为起始景点s,目标景点t,如果s与t之间连通,就选择一条道路使从s到达t的最小速度比最大,输出其最大边和最小边的比值的最小值的最简分数形式,若不连通,输出IMPOSSIBLE。
本题最小速度比是指一条道路上最大的速度与最小的速度的比值最小。且我们首先思考如何找到一条连通s与t道路上的最大边与最小边,怎么办?最小生成树!克鲁斯卡尔!
kruskal算法核心思想:
既然已经给出了邻接表。将道路按速度由小到大排序,枚举最小速度,初始视所有景点都为不连通,之后由最小速度的道路开始以速度从小到大枚举所有道路,判断道路两端的景点是否已经连通,若已经连通不做处理,若不连通则将该道路记录入最小生成树,标记道路两端为连通,判断s与t是否连通,若连通标记s与t可达,计算此时的最大速度与最小速度比值(因为道路速度由小到大遍历所以最小速度为开始遍历时的道路速度,最大速度为当前道路速度),将现在的比值与之前记录的比值比较,如果现在的值小于先前的值,将比值记录为现在的比值,并记录这时的最大速度与最小速度。
bool kruskal(int n, int m, int s, int t, int &maxL, int &minL){
//传入的n为景点数量,m为道路数量,s为起点,t为目标景点
//由于要改变maxL与minL记录最终的最大值与最小值,所以maxL与minL传引用
double ans = inf; //初始化比值为无穷大
bool flag = false; //标记s与t为不可达
sort(Edge + , Edge + + m, cmp); //将道路由小到大排序
for(int i = ; i <= m; i++){ //枚举最小速度
for(int k = ; k <= n; k++){ //初始化所有景点为不连通
father[k] = k;
}
int minlen = Edge[i].v, maxlen = ; //记录最小速度,初始化最大速度为0
for(int j = i; j <= m; j++){ //以速度由小到大枚举所有道路
int faNode1 = getFather(Edge[j].node1);
int faNode2 = getFather(Edge[j].node2);
maxlen = Edge[j].v; //记录最大速度为当前道路速度
if(faNode1 != faNode2){ //判断道路两端顶点是否连通
father[faNode1] = faNode2; //不连通就标记为连通
if(getFather(s) == getFather(t)){ //判断s与t是否连通
flag = true; //如果连通标记s与t为可达
if(ans > (double)maxlen / minlen){
//计算此时的最大速度与最小速度比值,将现在的比值与之前记录的比值比较
ans = (double)maxlen / minlen;
//如果现在的值小于先前的值,将比值记录为现在的比值
maxL = maxlen;
minL = minlen;
//记录这时的最大速度与最小速度
break;
}
}
}
}
}
if(flag){ //如果s与t可达返回true,否则返回false
return true;
}else{
return false;
}
}
kruskal
是否连通用并查集进行判断
int father[maxn], maxL, minL; //并查集部分
int getFather(int x)
{
if(father[x] == x)
return x;
else
return father[x] = getFather(father[x]);
}
并查集
之后得到了最小速度与最大速度,只需要让最大速度与最小速度除以他们的最大公约数便可以获得分子与分母。
AC代码
#include <bits/stdc++.h>
using namespace std;
const int inf = 0x7fffffff; //无穷大
const int maxn = 3e4+;
struct edge{ //edge保存道路
int node1, node2; //道路两端景点
int v; //道路速度
}Edge[maxn];
int gcd(int a, int b){ //计算最大公约数
if(b == )
return a;
else
return gcd( b, a % b);
}
bool cmp(edge e1, edge e2){ //道路按速度由小到大排序
return e1.v < e2.v;
}
int father[maxn], maxL, minL; //并查集部分
int getFather(int x)
{
if(father[x] == x)
return x;
else
return father[x] = getFather(father[x]);
}
bool kruskal(int n, int m, int s, int t, int &maxL, int &minL){
//传入的n为景点数量,m为道路数量,s为起点,t为目标景点
//由于要改变maxL与minL记录最终的最大值与最小值,所以maxL与minL传引用
double ans = inf; //初始化比值为无穷大
bool flag = false; //标记s与t为不可达
sort(Edge + , Edge + + m, cmp); //将道路由小到大排序
for(int i = ; i <= m; i++){ //枚举最小速度
for(int k = ; k <= n; k++){ //初始化所有景点为不连通
father[k] = k;
}
int minlen = Edge[i].v, maxlen = ; //记录最小速度,初始化最大速度为0
for(int j = i; j <= m; j++){ //以速度由小到大枚举所有道路
int faNode1 = getFather(Edge[j].node1);
int faNode2 = getFather(Edge[j].node2);
maxlen = Edge[j].v; //记录最大速度为当前道路速度
if(faNode1 != faNode2){ //判断道路两端顶点是否连通
father[faNode1] = faNode2; //不连通就标记为连通
if(getFather(s) == getFather(t)){ //判断s与t是否连通
flag = true; //如果连通标记s与t为可达
if(ans > (double)maxlen / minlen){
//计算此时的最大速度与最小速度比值,将现在的比值与之前记录的比值比较
ans = (double)maxlen / minlen;
//如果现在的值小于先前的值,将比值记录为现在的比值
maxL = maxlen;
minL = minlen;
//记录这时的最大速度与最小速度
break;
}
}
}
}
}
if(flag){ //如果s与t可达返回true,否则返回false
return true;
}else{
return false;
}
}
int main()
{
int numNode, numEdge;
while(scanf("%d%d", &numNode, &numEdge)!= EOF){ //输入景点数与道路数
for(int i = ; i <= numEdge; i++){ //输入道路
scanf("%d%d%d", &Edge[i].node1, &Edge[i].node2, &Edge[i].cost);
}
int s, t;
scanf("%d%d", &s, &t); //输入起点终点
int maxL = ;
int minL = ;
//初始化最大最小速度都为0
if(kruskal(numNode, numEdge, s, t, maxL, minL)){ //如果s与t连通
int temp = gcd(maxL, minL); //计算最大最小速度的最大公约数
//printf("%d\n", gcd(maxL, minL));
int a = maxL/temp, b = minL/temp; //计算分子分母
if(b != ) //分子不等于1输出最简分数
printf("%d/%d\n", a, b);
else
printf("%d\n", a); //分子为1直接输出分母
}else{ //不连通输出IMPOSSIBLE
printf("IMPOSSIBLE\n");
}
}
return ;
}
BZOJ1050 旅行comf(kruskal)的更多相关文章
- BZOJ-1050 旅行comf 并查集+乱搞
好久以前codevs上做过的,拿着改了改.. 1050: [HAOI2006]旅行comf Time Limit: 10 Sec Memory Limit: 162 MB Submit: 2194 S ...
- [BZOJ1050] [HAOI2006] 旅行comf (Kruskal, LCT)
Description 给你一个无向图,N(N<=500)个顶点, M(M<=5000)条边,每条边有一个权值Vi(Vi<30000).给你两个顶点S和T,求一条路径,使得路径上最大 ...
- [bzoj1050 HAOI2006] 旅行comf (kruskal)
传送门 Description 给你一个无向图,N(N<=500)个顶点, M(M<=5000)条边,每条边有一个权值Vi(Vi<30000).给你两个顶点S和T,求 一条路径,使得 ...
- 【BZOJ1050】[HAOI2006]旅行comf 并查集
[BZOJ1050][HAOI2006]旅行comf Description 给你一个无向图,N(N<=500)个顶点, M(M<=5000)条边,每条边有一个权值Vi(Vi<300 ...
- 【bzoj1050】[HAOI2006]旅行comf
1050: [HAOI2006]旅行comf Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2813 Solved: 1534[Submit][St ...
- bzoj1050【HAOI2006】旅行comf
1050: [HAOI2006]旅行comf Time Limit: 10 Sec Memory Limit: 162 MB Submit: 2205 Solved: 1174 [Submit][ ...
- 【BZOJ】【1050】【HAOI2006】旅行comf
枚举/暴力/Kruskal orz……我sb了……其实是sb题<_< 有一道题问的是最小极差生成树……(不记得是什么名字了,就是求最大边权与最小边权差最小的生成树)做法是枚举最小边,然后k ...
- 1050: [HAOI2006]旅行comf
1050: [HAOI2006]旅行comf Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1495 Solved: 737[Submit][Sta ...
- BZOJ 1050 [HAOI2006]旅行comf
1050: [HAOI2006]旅行comf Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1889 Solved: 976[Submit][Sta ...
随机推荐
- log4j打印MyBatis的sql语句配置
log4j.rootLogger=DEBUG,stdout log4j.appender.stdout=org.apache.log4j.ConsoleAppender log4j.appender. ...
- 使用Array.prototype.indexOf()的几点注意
对应indexOf这个方法,在日常开发中比较常见的应该是String.prototype.indexOf()方法,Array.prototype.indexOf()方法和其有很大的相似性,本文不想去描 ...
- jenkins启动失败,提示Starting Jenkins Jenkins requires Java8 or later, but you are running 1.7.0
# 背景 centos安装jenkins后,先启动jenkins服务,结果报错如下: 但自己明明已经安装了java8的 # 解决方法 既然安装了java8的话,那么证明是jenkins启动的是还是用的 ...
- PostgreSQL 区域设置
安装PostgreSQL 10.3 windows版本时区域请选择"default locale",安装成功后输入命令: show lc_ctype; show lc_collat ...
- linux进程管理(一)
进程介绍 程序和进程 程序是为了完成某种任务而设计的软件,比如OpenOffice是程序.什么是进程呢?进程就是运行中的程序. 一个运行着的程序,可能有多个进程. 比如自学it网所用的WWW服务器是a ...
- Sphinx全文检索
全文检索 一.生活中的数据总体分为: 结构化数据:指具有固定格式或有限长度的数据,如数据库,元数据等. 非结构化数据:指没有固定格式或不定长的数据,如邮件,word文档等. 非结构化数据还有一种叫法: ...
- Java50道经典习题-程序16 在控制台上打印九九乘法表
题目:输出9*9口诀.分析:利用双重for循环进行输出,分行与列考虑,共9行9列,i控制行,j控制列. public class Prog16 { public static void main(St ...
- “全栈2019”Java第一百零二章:哪些作用域可以声明局部内部类?
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- 【性能分析】使用Intel VTune Amplifier
本文转自 https://software.intel.com/zh-cn/blogs/2010/11/10/amplxe-cl/版权归原作者所有,如原作者有任何不允许转载之理由,本文将自行删除. I ...
- python接口自动化发送get请求 详解(一)
前言:接口自动化实现自动化脚本比较稳定,主要用到requests模块,后面我会把这个模块单独拉出来写一下. 一.环境安装 1.用pip安装requests模块 >>pip install ...