【[TJOI2017]城市】
好像\(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]城市】的更多相关文章
- 【BZOJ4890】[TJOI2017]城市(动态规划)
[BZOJ4890][TJOI2017]城市(动态规划) 题面 BZOJ 洛谷 题解 数据范围都这样了,显然可以暴力枚举断开哪条边. 然后求出两侧直径,暴力在直径上面找到一个点,使得其距离直径两端点的 ...
- [洛谷P3761] [TJOI2017]城市
洛谷题目链接:[TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速 ...
- 换根DP+树的直径【洛谷P3761】 [TJOI2017]城市
P3761 [TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公 ...
- bzoj4890[Tjoi2017]城市(树的半径)
4890: [Tjoi2017]城市 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 149 Solved: 91[Submit][Status][D ...
- [TJOI2017]城市(树的直径)
[TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公路相互可达, ...
- [BZOJ4890][TJOI2017]城市(DP)
题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公路相互可达,但是通过一条高速公路需要收 ...
- BZOJ4890 & 洛谷3761:[TJOI2017]城市——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=4890 https://www.luogu.org/problemnew/show/P3761 从加 ...
- [TJOI2017]城市 【树的直径+暴力+优化】
Online Judge:Luogu P3761 Label:树的直径,暴力 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有n座城市,n-1条高速公路,保证了 ...
- luogu P3761 [TJOI2017]城市 树的直径 bfs
LINK:城市 谢邀,学弟说的一道毒瘤题. 没有真正的省选题目毒瘤 或者说 写O(n)的做法确实毒瘤. 这里给一个花20min就写完的非常好写的暴力. 容易想到枚举哪条边删掉 删掉之后考虑在哪两个点上 ...
- [TJOI2017]城市
嘟嘟嘟 这题刚开始想复杂了,想什么dp去了,其实没那么难. 考虑断掉一条边,记分离出来的两棵子树为A和B,那么合并后的树的直径可能有三种情况: 1.A的直径. 2.B的直径 3.A的半径+边权+B的半 ...
随机推荐
- MySql——安装与配置与启动和停止
在Linux上安装mysql数据库,我们可以去其官网上下载mysql数据库的rpm包,http://dev.mysql.com/downloads/mysql/5.6.html#downloads,大 ...
- GreenPlum 大数据平台--web监控
一,安装web监控界面 01,准备 下载greenplum cc包,>> 解压缩 02,安装前配置 vim /greenplum/data/master/gpseg-/pg_hba.con ...
- Windows 7安装PlayReady出现“任务被禁用”错误信息
问题描述: Windows 7的Windows media center中安装PlayReady时出现:错误信息:任务被禁用.(异常来自 HRESULT:0x80041326) 解决办法: 先请确认是 ...
- git读书笔记以及使用技巧
[添加文件] git add 把文件修改添加到暂存区 git commit -m '' 把暂存区的所有内容提交到当前分支 [查看历史] git log 查看提交历史 git log -- ...
- 强哥的分享--如何使用Spring Boot做一个邮件系统
http://springboot.fun/ actuator是单机.集群环境下要使用Spring Boot Admin将各个单机的actuator集成越来 mvn clean package -Dm ...
- Windows server 搭建ftp服务器
1.安装ftp 2.端口端口21和20的入出端口 3.点击IIS 服务器证书 4.FTP界面上选择“FTP身份验证”——>“基本身份验证”–>”启用” 5.FTP界面选择 “FTP授权规则 ...
- 运算符重载关键字operator
operator关键字用来重载内置运算符,使用方法如下: public class OperatorController : Controller { // // GET: /Operator/ pu ...
- Git for Android Studio 学习笔记
http://learngitbranching.js.org/ 一个特别好的git学习教程 创建一个project,然后导入github
- PLC-Heart
- Linux漏洞分析入门笔记-Off-By-One(栈)
ubuntu-16.04.5(X86) IDA7.0 0x00.漏洞描述 1.什么是off by one?又称1字节溢出. 源字符串长度等于目标缓冲区长度时,将源字符串复制到目标缓冲区可能会导致off ...