[蓝桥杯][2013年第四届真题]大臣的旅费

链接

题干

很久以前,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最多花费的路费是多少。

样例输入

5
1 2 2
1 3 1
2 4 5
2 5 4

样例输出

135

算法

邻接表建树,求树的“直径”,即树中两点最远的距离;
可以通过根节点遍历出离根节点最远的节点a,以a节点再遍历出距离a节点最远的节点b,
a和b两点的距离即为“树的直径”

题解

#include<iostream>
#include<algorithm>
#include<cstring>

using namespace std;

const int N = 0x3f3f3f; 
struct node{
	int number;
	int next;
	int value;
}head[N];

bool st[N];
int h[N],n,ans,tx;
int idx;
void add(int a,int b,int c)
{
	head[idx].next =h[a];
	head[idx].number =b;
	head[idx].value = c;
	h[a] = idx++;
}

void dfs(int a,int sum)
{
	if(ans<sum)
	{
		ans = sum,tx = a;
	}
	st[a]= true;
	for(int i=h[a];i!=-1;i=head[i].next )
	{
		if(st[head[i].number ])continue; 
		st[head[i].number ] = true;
		dfs(head[i].number ,head[i].value + sum );
	}
}
int main()
{
	cin >> n;
	
	memset(h,-1,sizeof h);
	while(--n)
	{
		int a,b,c;
		cin >> a >> b >> c;
		add(a,b,c),add(b,a,c);
	}
	dfs(1,0);
	memset(st,false,sizeof st);
	dfs(tx,0);
	int s = ans*10+(ans+1)*ans/2;
	cout << s << endl;
	
	return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值