舒适的旅行(并查集)

1050: [HAOI2006]旅行comf

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 3107  Solved: 1701
[Submit][Status][Discuss]

Description

  给你一个无向图,N(N<=500)个顶点, M(M<=5000)条边,每条边有一个权值Vi(Vi<30000)。给你两个顶点S和T
,求一条路径,使得路径上最大边和最小边的比值最小。如果S和T之间没有路径,输出”IMPOSSIBLE”,否则输出
这个比值,如果需要,表示成一个既约分数。 备注: 两个顶点之间可能有多条路径。

Input

  第一行包含两个正整数,N和M。下来的M行每行包含三个正整数:x,y和v。表示景点x到景点y之间有一条双向
公路,车辆必须以速度v在该公路上行驶。最后一行包含两个正整数s,t,表示想知道从景点s到景点t最大最小速
度比最小的路径。s和t不可能相同。
1<N<=500,1<=x,y<=N,0<v<30000,0<M<=5000

Output

  如果景点s到景点t没有路径,输出“IMPOSSIBLE”。否则输出一个数,表示最小的速度比。如果需要,输出一
个既约分数。

Sample Input

【样例输入1】
4 2
1 2 1
3 4 2
1 4
【样例输入2】
3 3
1 2 10
1 2 5
2 3 8
1 3
【样例输入3】
3 2
1 2 2
2 3 4
1 3

Sample Output

【样例输出1】
IMPOSSIBLE
【样例输出2】
5/4
【样例输出3】
2

//很稳。。。根本不是搜索题,用搜索做肯定超时,做法是用并查集加个M^2的算法做

将边按权值排好序后,枚举边,依次将之作为最小边,再一重循环一直加入进来,直到s,t连通,那这条边一定是最大边

 1 #include <stdio.h>
 2 #include <iostream>
 3 #include <algorithm>
 4 #include <vector>
 5 using namespace std;
 6
 7 #define MAXM  5005
 8 #define INF  500000000
 9 typedef long long LL;
10
11 int n,m;
12 int s,t;
13 int f[505];
14 struct Bian
15 {
16     int from;
17     int to;
18     int v;
19     bool operator < (const Bian b) const
20     {return v<b.v;}
21 }G[MAXM];
22
23 int gcd(int a,int b)
24 {
25     return b==0?a:gcd(b,(a%b));
26 }
27
28 void Init_Bin()
29 {
30     for (int i=1;i<=n;i++)
31         f[i]=i;
32 }
33
34 int find_h(int x)
35 {
36     if (x!=f[x])
37         f[x]=find_h(f[x]);
38     return f[x];
39 }
40
41
42 int main()
43 {
44     while (scanf("%d%d",&n,&m)!=EOF)
45     {
46         for (int i=0;i<m;i++)
47         {
48             int x,y,v;
49             scanf("%d%d%d",&x,&y,&v);
50             G[i]=(Bian){x,y,v};
51         }
52         sort(G,G+m);
53         scanf("%d%d",&s,&t);
54         int ansx=INF,ansy=-INF;
55         double res=INF;
56         for (int i=0;i<m;i++)
57         {
58             if (i>0&&G[i].v==G[i-1].v) continue;//剪枝,减很久时间
59             Init_Bin(); //初始化并查集
60             int low =G[i].v,high;
61             for (int j=i;j<m;j++)
62             {
63                 int fx=find_h(G[j].from),fy=find_h(G[j].to);
64                 if (fx!=fy)
65                     f[fx]=f[fy];
66                 high=G[j].v;
67                 if (find_h(s)==find_h(t))
68                 {
69                     double kk=(double)high/low;
70                     if (kk<res)
71                     {
72                         res=kk;
73                         ansx=high;
74                         ansy=low;
75                     }
76                     break;
77                 }
78             }
79         }
80         if (res==INF)
81             printf("IMPOSSIBLE\n");
82         else
83         {
84             int yue=gcd(ansx,ansy);
85             if (ansy/yue==1)
86                 printf("%d\n",ansx/yue);
87             else
88                 printf("%d/%d\n",ansx/yue,ansy/yue);
89         }
90     }
91     return 0;
92 }

时间: 2024-10-10 01:15:36

舒适的旅行(并查集)的相关文章

BZOJ 1050 旅行(并查集)

很好的一道题. 首先注意,要使的s到t的路径上最大边/最小边的值最小.我们可以尝试一下二分并验证答案. 但是不能二分最大边/最小边. 我们可以二分 最小边的权值.于是算法就出来了. 首先把边权排序.然后枚举最小的边,再依次添加不小于该边的边,直到s和t联通.用并查集维护即可. # include <cstdio> # include <cstring> # include <cstdlib> # include <iostream> # include &l

并查集+二分 旅行

题目描述: A国有n座城市,每座城市都十分美,这使得A国的民众们非常喜欢旅行.然而,A国的交通十分落后,这里只有m条双向的道路,并且这些道路都十分崎岖,有的甚至还是山路,只能靠步行.通过每条道路的长度.泥泞程度等因素,我们给每条道路评估一个"崎岖度",表示通过这条道路的不舒适程度. 从X城市经过若干条道路到达Y城市,我们称这次旅行的"代价"为所经过道路"崎岖度"的最大值.当然,如果从X城市到Y城市有多条路线,民众们会自觉选择"代价&qu

BZOJ 1050 旅行comf(枚举最小边-并查集)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1050 题意:给出一个带权图.求一条s到t的路径使得这条路径上最大最小边的比值最小? 思路:将边排序.枚举最小边,然后将边一个一个插到并查集里,s和t联通时计算更新答案. struct node { int u,v,w; void get() { RD(u,v,w); } }; int cmp(node a,node b) { return a.w<b.w; } int n,m,s,t;

codevs 1001 舒适的路线 (并查集)

题目描述 Description Z小镇是一个景色宜人的地方,吸引来自各地的观光客来此旅游观光. Z小镇附近共有 N(1<N≤500)个景点(编号为1,2,3,…,N),这些景点被M(0<M≤5000)条道路连接着,所有道路都是双向的,两个景点之间可能有多条道路.也许是为了保护该地的旅游资源,Z小镇有个奇怪的规定,就是对于一条给定的公路Ri,任何在该公路上行驶的车辆速度必须为Vi.频繁的改变速度使得游客们很不舒服,因此大家从一个景点前往另一个景点的时候,都希望选择行使过程中最大速度和最小速度的

nyoj 711最舒适的路线(第六届河南省程序设计大赛 并查集)

最舒适的路线 时间限制:5000 ms  |  内存限制:65535 KB 难度:5 描述 异形卵潜伏在某区域的一个神经网络中.其网络共有N个神经元(编号为1,2,3,-,N),这些神经元由M条通道连接着.两个神经元之间可能有多条通道.异形卵可以在这些通道上来回游动,但在神经网络中任一条通道的游动速度必须是一定的.当然异形卵不希望从一条通道游动到另一条通道速度变化太大,否则它会很不舒服. 现在异形卵聚居在神经元S点,想游动到神经元T点.它希望选择一条游动过程中通道最大速度与最小速度比尽可能小的路

BZOJ 1050: [HAOI2006]旅行comf(枚举+并查集)

[HAOI2006]旅行comf Description 给你一个无向图,N(N<=500)个顶点, M(M<=5000)条边,每条边有一个权值Vi(Vi<30000).给你两个顶点S和T,求一条路径,使得路径上最大边和最小边的比值最小.如果S和T之间没有路径,输出”IMPOSSIBLE”,否则输出这个比值,如果需要,表示成一个既约分数. 备注: 两个顶点之间可能有多条路径. Input 第一行包含两个正整数,N和M.下来的M行每行包含三个正整数:x,y和v.表示景点x到景点y之间有一条

bzoj 1050: [HAOI2006]旅行comf(最小生成树+并查集)

1050: [HAOI2006]旅行comf Time Limit: 10 Sec  Memory Limit: 162 MB Submit: 2405  Solved: 1282 [Submit][Status][Discuss] Description 给你一个无向图,N(N<=500)个顶点, M(M<=5000)条边,每条边有一个权值Vi(Vi<30000).给你两个顶点S和T ,求一条路径,使得路径上最大边和最小边的比值最小.如果S和T之间没有路径,输出"IMPOSS

枚举+并查集 之 CODE[VS] 1001 舒适的路线 2006年

/* 枚举所有情况,通过并查集判断在当前情况下,是否包含起始和终止点, 若包含,则需判断是否在当前情况下,最大速度和最小速度的比更小, 若是,则更新最大速度和最小速度. 最后枚举过所有情况后,即获得答案. */ 1 #include <iostream> 2 #include <cstdlib> 3 #include <cstdio> 4 #include <cstddef> 5 #include <iterator> 6 #include &l

CODEVS1001 舒适的路线 (并查集)

对所有边从大到小排序,枚举最大边,O(m)验证,用并查集维护图是否联通. program CODEVS1001; const maxm=5008; maxn=508; INF=2000000000; type arr=record u,v,w:int64; end; var eg:array[0..maxm] of arr; fa:array[0..maxn] of longint; i,j,m,n,x,y,z,s,t,u,v,d:longint; ans1,ans2:int64; proced