很久以前,T王国空前繁荣。为了更好地管理国家,王国修建了大量的快速路,用于连接首都和王国内的各大城市。
为节省经费,T国的大臣们经过思考,制定了一套优秀的修建方案,使得任何一个大城市都能从首都直接或者通过其他大城市间接到达。同时,如果不重复经过大城市,从首都到达每个大城市的方案都是唯一的。
J是T国重要大臣,他巡查于各大城市之间,体察民情。所以,从一个城市马不停蹄地到另一个城市成了J最常做的事情。他有一个钱袋,用于存放往来城市间的路费。
聪明的J发现,如果不在某个城市停下来修整,在连续行进过程中,他所花的路费与他已走过的距离有关,在走第x千米到第x+1千米这一千米中(x是整数),他花费的路费是x+10这么多。也就是说走1千米花费11,走2千米要花费23。
J大臣想知道:他从某一个城市出发,中间不休息,到达另一个城市,所有可能花费的路费中最多是多少呢?
输入的第一行包含一个整数n,表示包括首都在内的T王国的城市数
城市从1开始依次编号,1号城市为首都。
接下来n-1行,描述T国的高速路(T国的高速路一定是n-1条)
每行三个整数Pi, Qi, Di,表示城市Pi和城市Qi之间有一条高速路,长度为Di千米。
输出一个整数,表示大臣J最多花费的路费是多少。
1 2 2
1 3 1
2 4 5
2 5 4
大臣J从城市4到城市5要花费135的路费。
在去年十二月跟这道题死磕过,也看了提示,但是就是不理解,没想到今天再见这道题,居然明白提示的意思了,发现这道题其实挺简单.
题目实际上给出了一个最小生成树,一点到另外一点的路径是唯一的,于是:
设在最长路径的两端为点a和点b, 则其余点若要构成以自身为一端的最远路径,另一端点非a即b.
用反证法:
设端点x与c(c != a && c != b)构成以x为一端的最远路径,即x到c大于x到a和x到b,则有a到c和b到c都大于a到b,这与前提假设不符,故x必与a,b中的一点构成以x为一端的最长路径.
- #include<stdio.h>
- #include<stdlib.h>
- struct Bus
- {
- int n;
- short **map;
- int *foot;
- int max;
- int fast;
- } bus;
- void dfs(int now, int value)
- {
- bus.foot[now] = 1;
- int sign = 0;
- int i;
- for(i = 0; i < bus.n; i++)
- {
- int t = bus.map[now][i];
- if(bus.foot[i] == 0 && t > 0)
- {
- sign = 1;
- dfs(i, value + t);
- }
- }
- if(sign == 0 && value > bus.max)
- {
- bus.max = value;
- bus.fast = now;
- }
- bus.foot[now] = 0;
- }
- int main(void)
- {
- int i;
- scanf("%d", &bus.n);
- bus.map = calloc(bus.n, sizeof(short *));
- short *p = calloc(bus.n * bus.n, sizeof(short));
- for(i = 0; i < bus.n; i++)
- {
- bus.map[i] = p + bus.n * i;
- }
- bus.foot = calloc(bus.n, sizeof(int));
- for(i = 0; i < bus.n - 1; i++)
- {
- short a, b, c;
- scanf("%hd%hd%hd", &a, &b, &c);
- bus.map[a - 1][b - 1] = bus.map[b - 1][a - 1] = c;
- }
- bus.max = 0;
- dfs(0, 0);
- int thePoint = bus.fast;
- bus.max = 0;
- dfs(thePoint, 0);
- printf("%d", bus.max * (bus.max + 1) / 2 + bus.max * 10);
- return 0;
- }
本文探讨了一个关于在特定道路网络中寻找两节点间最长路径的问题,并计算沿此路径旅行时累积的最大路费。通过分析给出的最小生成树,利用深度优先搜索算法找到网络中最长的路径,并结合费用计算公式得出最终答案。
608

被折叠的 条评论
为什么被折叠?



