题目

好像\(noip\)之前做某雅礼的题的时候看到过这道题的数据范围增强版

当时那道题数据范围是\(3e5\)感觉神仙的一批

这道题数据范围\(5e3\)那岂不是可以\(O(n^2)\)水过

有一点非常显然就是我们断开的那条边肯定是树的直径上的一条边,于是我们可以先来两遍树形\(dp\)求出子树内最长链,次长链,和子树外最长链

这个时候树的直径就知道啦

我们枚举断边,当然只断直径上的边

之后得到了两个联通块,我们对这两个联通块求一下直径

考虑连接哪两个点会使得新直径尽量小

显然分别从这两个联通块里找到一个点,这个点到联通块内最远的点距离最小

新直径就是\(max\{r_1,r_2,w+d_1+d_2\}\),\(r_1,r_2\)是两个联通块的直径,\(w\)是断开的边的边权,\(d_1,d_2\)是联通块内最远的点距离最小的距离

代码

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#define re register
#define maxn 5005
#define LL long long
inline int max(int a,int b) {return a>b?a:b;}
inline int min(int a,int b) {return a<b?a:b;}
inline int read()
{
char c=getchar();int x=0;
while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
struct E{int v,nxt,w;}e[maxn<<1];
int head[maxn],f[maxn],g[maxn],t[maxn],T[maxn],F[maxn],dp[maxn];
int n,num,R,ans,root;
inline void add(int x,int y,int z) {e[++num].v=y;e[num].w=z;e[num].nxt=head[x];head[x]=num;}
void dfs(int x,int fa)
{
f[x]=g[x]=0;
for(re int i=head[x];i;i=e[i].nxt)
if(e[i].v!=fa)
{
if(!fa&&root==e[i].v) continue;
dfs(e[i].v,x);
if(f[e[i].v]+e[i].w>f[x]) g[x]=f[x],f[x]=f[e[i].v]+e[i].w;
else if(f[e[i].v]+e[i].w>g[x]) g[x]=f[e[i].v]+e[i].w;
else if(g[e[i].v]+e[i].w>g[x]) g[x]=g[e[i].v]+e[i].w;
}
}
void redfs(int x,int fa)
{
for(re int i=head[x];i;i=e[i].nxt)
if(e[i].v!=fa)
{
if(!fa&&root==e[i].v) continue;
t[e[i].v]=t[x]+e[i].w;
if(f[e[i].v]+e[i].w==f[x]) t[e[i].v]=max(t[e[i].v],g[x]+e[i].w);
else t[e[i].v]=max(t[e[i].v],f[x]+e[i].w);
redfs(e[i].v,x);
}
dp[x]=max(f[x],t[x]);
}
int find(int x,int fa)
{
int k=dp[x];
for(re int i=head[x];i;i=e[i].nxt)
if(e[i].v!=fa)
{
if(!fa&&root==e[i].v) continue;
k=min(k,find(e[i].v,x));
}
return k;
}
void check(int x,int fa)
{
for(re int i=head[x];i;i=e[i].nxt)
if(e[i].v!=fa)
{
if(T[e[i].v]+F[e[i].v]==R)
{
root=e[i].v;
memset(t,0,sizeof(t));
dfs(e[i].v,x);redfs(e[i].v,x);
dfs(x,0);redfs(x,0);
int now=0;
for(re int j=1;j<=n;++j) now=max(now,f[j]+g[j]);
now=max(now,e[i].w+find(e[i].v,x)+find(x,0));
ans=min(ans,now);
}
check(e[i].v,x);
}
}
int main()
{
n=read();int x,y,z;
for(re int i=1;i<n;i++) x=read(),y=read(),z=read(),add(x,y,z),add(y,x,z);
dfs(1,0),redfs(1,0);
for(re int i=1;i<=n;i++) R=max(R,f[i]+g[i]);ans=R;
for(re int i=1;i<=n;i++) T[i]=t[i],F[i]=f[i];
check(1,0);
printf("%d\n",ans);
return 0;
}

【[TJOI2017]城市】的更多相关文章

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

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

  2. [洛谷P3761] [TJOI2017]城市

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

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

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

  4. bzoj4890[Tjoi2017]城市(树的半径)

    4890: [Tjoi2017]城市 Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 149  Solved: 91[Submit][Status][D ...

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

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

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

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

  7. BZOJ4890 & 洛谷3761:[TJOI2017]城市——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4890 https://www.luogu.org/problemnew/show/P3761 从加 ...

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

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

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

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

  10. [TJOI2017]城市

    嘟嘟嘟 这题刚开始想复杂了,想什么dp去了,其实没那么难. 考虑断掉一条边,记分离出来的两棵子树为A和B,那么合并后的树的直径可能有三种情况: 1.A的直径. 2.B的直径 3.A的半径+边权+B的半 ...

随机推荐

  1. spark第二篇:Application Submission Guide

    提交应用 Spark的bin目录中的spark-submit脚本用于启动集群上的应用程序.它可以通过一个统一的接口使用所有Spark支持的集群管理器. 绑定应用程序的依赖 如果你的代码依赖其他项目,你 ...

  2. rancher初级(搭建+基本操作+web应用部署)

    Rancher搭建 首先rancher需要安装了docker的linux环境,我的系统版本为 在docker的基础上启动rancher服务器,Rancher 服务器是一个 Docker image,所 ...

  3. linux 期中架构之 nginx 安装与排错

    1, 安装 nginx 所需要的pcre库 即:perl 兼容正则表达式 yum install pcre pcre-devel -y rpm -qa pcre pcre-devel 检查是否安装好p ...

  4. Oracle 数据库和Sql Server数据库的区别

    Oracle数据库的访问方式,和SqlServer数据库是有很大差别的,下面用图来说明: 1.Sql Server数据库 SqlServer数据库的访问方式,大致是:假设用户通过sa登录SqlServ ...

  5. ionic 打包安卓包 (debug调试版和 release发布版)

    一.配置环境: 先按照之前的文章,配置好环境需要: 安装jdk,配置环境变量:( http://www.cnblogs.com/loveyaxin/p/7520618.html) 安装android- ...

  6. Java学习第二十五天

    1:如何让Netbeans的东西Eclipse能访问. 在Eclipse中创建项目,把Netbeans项目的src下的东西给拿过来即可. 注意:修改项目编码为UTF-8 2:GUI(了解) (1)用户 ...

  7. JavaScript对象 原型

    javascript对象就是一组数据和功能的集合,除原始类型(string.number.boolean.null.undefined)之外,其余都是对象. 可以通过对象直接量(字面量).new.和O ...

  8. View视图调用控制器方法

    1.@using XXX.Controllers;//引用控制器 2. var otherController = DependencyResolver.Current.GetService<U ...

  9. C# 创建一个WCF服务

    做代码统计,方便以后使用: app.config配置文件设置: <configuration> <system.serviceModel> <bindings> & ...

  10. 从零开始的全栈工程师——html篇1.3

    文本.字体css样式与前期英语单词汇总 一.文本样式(text) 1.颜色:color:red; 2.文本对齐方式:text-align:left/center/right/justify; left ...