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 ...
随机推荐
- vs2008快捷键极其技巧
vs2008快捷键极其技巧 1. 工具: Microsoft Visual Studio 2008 Version 9.0.21022.8 RTM Microsoft .NET Framework V ...
- centos 7.6 开机报错信息(一):welcome to emergency mode!
welcome to emergency mode!after logging in ,type "journalctl -xb" to view system logs,&quo ...
- scvmm sdk之powershell(一)
shell表示计算机操作系统中的壳层,与之相对的是内核,内核不能与用户直接交互,而是通过shell为用户提供操作界面,shell分为两类,一种提供命令行界面,一种提供图形界面.windows powe ...
- 记一次很坑的python2与python3共存问题
当添加PYTHONPATH环境变量时,无论输入pip2 -V还是pip3 -V都显示的是python2的环境变量,使用pip3 install 时也是安装在了python2的三方库(因为python2 ...
- ES6摘抄
1.函数可选参数function log(x, y = 'World') {} 只能作为尾参数使用,因为如果不是尾参数还是要输入的.2.函数参数默认值与解构赋值结合使用.(注意对象冒号解构等号)fun ...
- Monthly数据类型
Monthly由来 最近在做关于智能财税的项目,大量用到了账期相关的数据操作.项目已有两年历史了,对于账期数据,前辈们用的是DateTime数据类型,即每个月的最后一天就是账期.而用DateTime来 ...
- Rx.net 例子——(1)基础
1.订阅序列 using System; using System.Collections.Generic; using System.Linq; using System.Text; using S ...
- .Net Core使用HttpClient请求Web API注意事项
HttpClient 使用HttpClient可以很方便的请求Web API,但在使用时有一些需要注意的地方,不然会给你的程序带来毁灭性的问题. HttpClient是一个继承了IDisposable ...
- iOS Facebook SDK
iOS 使用 Facebook SDK 可以登录,分享,发布通知(Notifications)等. 首先下载 Facebook SDK.然后在 Facebook Developer 上注册自己的 ap ...
- sgi stl内存池实现------源码加翻译
class __default_alloc_template { enum { unit = 8 };//分配单位 后面直接用8代替 enum { max_bytes = 128 };//最大分配字节 ...