4890: [Tjoi2017]城市

Time Limit: 30 Sec  Memory Limit: 128 MB
Submit: 149  Solved: 91
[Submit][Status][Discuss]

Description

 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作。这个地区一共有ri座城市,《-1条高速公路,保证了任意两运城市之间都可以通过高速公路相互可达,但是通过一条高速公路需要收取一定的交通费用。小明对这个地区深入研究后,觉得这个地区的交通费用太贵。小明想彻底改造这个地区,但是由于上司给他的资源有限,因而小明现在只能对一条高速公路进行改造,改造的方式就是去掉一条高速公路,并且重新修建一条一样的高速公路(即交通费用一样),使得这个地区的两个城市之间的最大交通费用最小(即使得交通费用最大的两座城市之间的交通费用最小),并且保证修建完之后任意两座城市相互可达。如果你是小明,你怎么解决这个问题?

Input

输入数据的第一行为一个整数n,代表城市个数。

接下来的n - 1行分别代表了最初的n-1条公路情况。每一行都有三个整数u,v,d。u,v代表这条公路的两端城市标号,d代表这条公路的交通费用。

Output

 输出数据仅有一行,一个整数,表示进行了最优的改造之后,该地区两城市 之间最大交通费用。

Sample Input

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

Sample Output

 7

HINT

对于30%的数据,1<=n<500

对于100%的数据,1<=n<=5000

1 <= u,v <= n,1<= d <= 2000

/*
要想删一条边再添一条边使一棵树中两点间最大距离最小
容易想到跟树的直径有关
N比较小,可以枚举要删那一条边。删掉(u,v)之后树就成了两个联通块。
如果树的直径没变,对答案没有影响。 可以求出联通块1,2的直径。
(所以可以只枚举删直径上的边,但我没有...)
那么问题就转化为在1,2联通块内分别找一个点使它到联通块内最远的点距离最近。
这个距离就是树的半径。然后把这两个点连起来即可。
如何维护半径?
求直径时需要求出每个点为起点的最长链和次长链
考虑距离它最远的那个点,在它的子树内还是子树外
子树内:最长链
子树外:dfs维护这个点子树外的最长链
如何维护一个点子树外的最长链?
当dfs到一个x时,对于x和他的父亲y,若x不在y的最长链内,那么ans[x](x到子树外最远距离)为max(y最长链+dis[x][y],ans[y]+dis[x][y])
否则ans[x]为max(y的次长链+dis[x][y],ans[y]+dis[x][y])
最后对每种情况答案取min即可。
*/
#include<bits/stdc++.h> #define N 5007
#define inf 0x3f3f3f3f using namespace std;
int n,m,ans,cnt,dis,res;
int head[N<<],mv[N],u[N],v[N],w[N];
int dp[N][];
bool vis[N];
struct edge{
int u,v,w,nxt;
}e[N<<]; inline void add(int u,int v,int w)
{
e[++cnt].v=v;e[cnt].w=w;e[cnt].nxt=head[u];head[u]=cnt;
} inline int read()
{
int x=,f=;char c=getchar();
while(c>''||c<''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
} void getd(int u)
{
for(int i=head[u];i;i=e[i].nxt)
{
int v=e[i].v;
if(vis[v]) continue;
vis[v]=;getd(v);
if(dp[u][]<dp[v][]+e[i].w)
{
dp[u][]=dp[u][];mv[u]=v;
dp[u][]=dp[v][]+e[i].w;
}
else if(dp[u][]<dp[v][]+e[i].w)
dp[u][]=dp[v][]+e[i].w;
}dis=max(dis,dp[u][]+dp[u][]);
} void getr(int u,int from)
{
res=min(res,max(from,dp[u][]));
for(int i=head[u];i;i=e[i].nxt)
{
int v=e[i].v;
if(!vis[v]) continue;
vis[v]=;
if(mv[u]==v)
getr(v,max(dp[u][]+e[i].w,from+e[i].w));
else
getr(v,max(dp[u][]+e[i].w,from+e[i].w)); }
} void clear()
{
memset(dp,,sizeof dp);
memset(mv,,sizeof mv);
memset(vis,,sizeof vis);
res=inf;dis=;
} int main()
{
n=read();
for(int i=;i<n;i++)
{
u[i]=read();v[i]=read();w[i]=read();
add(u[i],v[i],w[i]);add(v[i],u[i],w[i]);
}
int d1,d2,r1,r2;ans=res=inf;dis=;
for(int i=;i<n;i++)
{
vis[v[i]]=;getd(u[i]); d1=dis;
dis=;getd(v[i]); d2=dis;
//联通块1,2的直径
vis[v[i]]=;getr(u[i],); r1=res;
res=inf;getr(v[i],); r2=res;
//联通块1,2的半径
ans=min(ans,max(max(d1,d2),r1+r2+w[i]));
clear();
}
printf("%d\n",ans);
return ;
}

bzoj4890[Tjoi2017]城市(树的半径)的更多相关文章

  1. BZOJ4890 Tjoi2017城市

    显然删掉的边肯定是直径上的边.考虑枚举删哪一条.然后考虑怎么连.显然新边应该满足其两端点在各自树中作为根能使树深度最小.只要线性求出这个东西就可以了,这与求树的重心的过程类似. #include< ...

  2. [BZOJ4890][TJOI2017]城市(DP)

    题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公路相互可达,但是通过一条高速公路需要收 ...

  3. BZOJ4890 [Tjoi2017]城市 【树形dp】

    题目链接 BZOJ4890 题解 枚举断开哪一条边,然后对剩余的两棵树分别做一遍换根法树形dp 需要求出每个点到树中其它点距离的最大值\(f[i]\)和次大值\(g[i]\)[用以辅助换根计算最大值] ...

  4. [TJOI2017] 城市 (树的直径,贪心)

    题目链接 Solution 这道题,调了我一晚上... 一直80分 >_<|| ... 考虑到几点: 分开任意一条边 \(u\) ,那么其肯定会断成两棵树. 肯定是分开直径上的边最优,否则 ...

  5. luogu P3761 [TJOI2017]城市 树的直径 bfs

    LINK:城市 谢邀,学弟说的一道毒瘤题. 没有真正的省选题目毒瘤 或者说 写O(n)的做法确实毒瘤. 这里给一个花20min就写完的非常好写的暴力. 容易想到枚举哪条边删掉 删掉之后考虑在哪两个点上 ...

  6. 【BZOJ4890】[TJOI2017]城市(动态规划)

    [BZOJ4890][TJOI2017]城市(动态规划) 题面 BZOJ 洛谷 题解 数据范围都这样了,显然可以暴力枚举断开哪条边. 然后求出两侧直径,暴力在直径上面找到一个点,使得其距离直径两端点的 ...

  7. [TJOI2017]城市(树的直径)

    [TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公路相互可达, ...

  8. 换根DP+树的直径【洛谷P3761】 [TJOI2017]城市

    P3761 [TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公 ...

  9. [TJOI2017]城市 【树的直径+暴力+优化】

    Online Judge:Luogu P3761 Label:树的直径,暴力 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有n座城市,n-1条高速公路,保证了 ...

随机推荐

  1. PS注意点

    2.颜色 设计师应该具备审美能力. 3.实验 不断的练习会让你学习到更多的东西,请不要给自己太多压力,你的付出不会仅仅只让你原地踏步,要坚持.   填充和不透明的掌握. 还有流量的使用.   填充是一 ...

  2. C51 独立按键 个人笔记

    独立按键类似于一个开关,按下时开关闭合 防抖 硬件防抖 软件防抖 通过延时,滤掉抖动的部分 电路图 普中科技的开发板,独立按键电路图如下 判断按键按下 因此判断是否按下开关的方法是看引脚是否为低电平( ...

  3. [ZJOI2010] 数字统计

    [ZJOI2010] 数字统计 题目 给定两个正整数a和b,求在[a,b]中的所有整数中,每个数码(digit)各出现了多少次. INPUT 输入文件中仅包含一行两个整数a.b,含义如上所述 OUTP ...

  4. 光纤通信(codevs 1955)

    题目描述 Description 农民John 想要用光纤连通他的N (1 <= N <= 1,000)个牲口棚(编号1..N).但是,牲口棚位于一个大池塘边,他仅可以连通相邻的牲口棚.J ...

  5. cdq分治入门--BZOJ3262: 陌上花开

    n<=100000个人,每个人三个属性Ai,Bi,Ci,一个人i的等级为Ai>=Aj,Bi>=Bj,Ci>=Cj的人数,求每个等级有多少人. 裸的三维偏序.按照常规思路,一维排 ...

  6. Python基础之 一 字典(dict)

    字典:是一种key - value的数据类型.语法:info = { key:value }特性:无序,key必须唯一(所以天生去重) 方法如下:del dict[key]:删除字典指定键len(di ...

  7. 2017 CCPC 杭州 HDU6265B 积性函数

    题目链接 http://acm.hdu.edu.cn/downloads/CCPC2018-Hangzhou-ProblemSet.pdf B题 数论题      h(n)=∑ d|n φ(d) × ...

  8. Codeforces 645A Amity Assessment【八数码】

    题目链接: http://codeforces.com/problemset/problem/645/A 题意: 2*2的八数码问题 分析: 这题n为2,不需要搜索,直接判断字母排列顺序就好了. 注意 ...

  9. [bzoj3306]树_dfs序_线段树_倍增lca

    树 bzoj-3306 题目大意:给定一颗n个节点的树,支持换根.修改点权.查询子树最小值. 注释:$1\le n,q\le 10^5$. 想法: 如果没有换根操作,就是$dfs$序+线段树维护区间最 ...

  10. poj——1469 COURSES

    COURSES Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 24192   Accepted: 9426 Descript ...