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 ...
随机推荐
- ubuntu16.04 安装jdk 错误解决
错误 $ apt-get install openjdk-9-jdk Errors were encountered while processing: /var/cache/apt/archives ...
- CHARPTER 3--INDEX DMVs
1.查找最重要的缺失的索引 --======================================================= --查找最重要的缺失的索引 ) DB_NAME() AS ...
- leetcode 字符串转整数(atoi)
实现atoi,将字符串转为整数. 在找到第一个非空字符之前,需要移除掉字符串中的空格字符.如果第一个非空字符是正号或负号,选取该符号,并将其与后面尽可能多的连续的数字组合起来,这部分字符即为整数的值. ...
- how to trace the error log
Executed as user: WTC\Ebw.Admin. Transaction (Process ID 95) was deadlocked on lock resources with a ...
- Plasma Cash合约解读
Plasma Cash合约解读 SmartPlasma 合约解读 1. 合约代码 2. 合约文件简单介绍 3. Plasma Cash 的基础数据结构 3.1 Plasma Cash 中的资产 3.2 ...
- 六、linux目录结构知识
1.显示行号: cat -n 2.set nu 3.tail -f a.txt 查看文件的尾部变化 4.w 当前的登陆用户 5.yum包管理工具底层调用的还是 rpm -ivh 包名 ...
- 蹭你wifi后 我竟然干了这样的事
前言:故事发生在前两天,我们去参观工业园区内一家电商公司. 去参观他们公司的时候,我说要用下无线网,他们技术说密码就是他们的网站域名,我一脸懵逼表示我不知道域名,然后对方接过我手机给我连上了他们wif ...
- sublime package control失败
https://blog.csdn.net/jyfu2_12/article/details/86667132 安装时的错误提示如下: Error installing Package Control ...
- nginx的几种负载均衡策略
转自https://www.cnblogs.com/1214804270hacker/p/9325150.html 一.关于Nginx的负载均衡 在服务器集群中,Nginx起到一个代理服务器的角色(即 ...
- SpringMvc @RequestMapping原理
讲这个之前,我们得先知道在SpringMvc启动时,会加载所有的Bean类,就是加了@Controller,@Component等组件标识的类,然后会把@RequestMapping的方法也加入到一个 ...