好久没写题解了。这题不算太水就写一下题解。

话说回来,虽然不水但是挺裸。可以说题意即一半题解了。

我猜粘了题面也没有人去看的,所以直接人话题意了。


给一棵树,点数1e6,(当年noip的n当然是只有300了,,),就管他叫树网。

首先定义树的直径:树上最远点对之间的路径。我们定义树的一个点到一段路径的距离是:点和路径上最近的点之间路径长。

然后定义一段路径的偏心距ecc:除了这这路径上的点,其他点到这条路径的距离中的max。(和所有点没区别)

现在要求出这样一个路径,它在一条直径上(直径可能不止一个),它的长度不超过一个给定的值lim,满足前两个条件的情况下偏心距最小。我们叫他树网的核。(终于扣题了woc)


所以我们需要知道一些事情。

1,树的直径可能不止一个,但不论用哪个直径所能求出的该最小偏心距相等。

2,离一个点最远的点一定是一条直径的一端。

虽然我不知道为啥。。但我觉得很有道理啊对不对。。。

于是在这基础上思路就很明显。

先根据性质2求出一条直径。然后显然是不超过lim情况下这一段越长越好。于是枚举结束点,开始点随之后移(就是双指针啥的扫一遍)。

对于一条路径其偏心距有两种情况:

1,开始点或结束点与所在直径上离得近的那个端点的距离。

2,不经过所在直径的情况下,能到达的其他点中与它距离最远的距离。

仔细想想(YY)一下就知道,不理解,,,私吧。

情况2与起始点结束点无关,可以预处理f(x),复杂度是O(n)的,因为一个点只会被遍历一次。

于是情况2变成了单调队列问题:求区间最大值,,不是定长区间了,deque的pop_front条件变为了q.front()与当前枚举的结束点距离<lim。

要注意的是q.front()并不是起始点,而是起始点到结束点中f(x)最大的。(好像只有我这么想脑残wa了一发)起始点是贪心选的,因为越长越好,只要不超过lim就好。

貌似没有什么细节问题了,dfs还是巨好写的。附代码:

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<queue>
using namespace std;
const int N=;
inline int read(){
int r=,c=getchar();
while(!isdigit(c))c=getchar();
while(isdigit(c))
r=r*+c-'',c=getchar();
return r;
}
struct Edge{
int to,nxt,w;
}e[N*];
int head[N],cnt=;
void add(int u,int v,int w){
e[cnt]=(Edge){v,head[u],w};
head[u]=cnt++;
e[cnt]=(Edge){u,head[v],w};
head[v]=cnt++;
}
int n,lim,mx;
int S,T,d[N];
bool ind[N];
void dfs(int u,int fa){
mx=max(mx,d[u]);
for(int i=head[u];i;i=e[i].nxt){
int v=e[i].to;
if(v==fa||ind[v])continue;
d[v]=d[u]+e[i].w;
dfs(v,u);
}
}
int nxt[N],dis[N];
void dfs2(int u,int fa){
for(int i=head[u];i;i=e[i].nxt){
int v=e[i].to;
if(v==fa)continue;
dfs2(v,u);
if(ind[v])
nxt[u]=v,dis[u]=e[i].w,ind[u]=;
}
}
void init(){
n=read(),lim=read();
for(int i=;i<n;i++){
int u=read(),v=read(),w=read();
add(u,v,w);
}
}
void diameter(){
mx=;dfs(,);
for(int i=;i<=n;i++)
if(d[i]==mx)S=i;
memset(d,,sizeof d);
mx=;dfs(S,);
for(int i=;i<=n;i++)
if(d[i]==mx)T=i;
}
int l[N],f[N];
void get(){
ind[T]=;dfs2(S,);
int t=;
for(int u=S;u;u=nxt[u]){
d[u]=mx=;l[u]=t;
t+=dis[u];
dfs(u,);f[u]=mx;
}
}
void solve(){
deque<int>q;
q.push_back(S);
int ans=2e9,h=S;
for(int u=S;u;u=nxt[u]){
int m=f[u];
while(!q.empty()&&f[q.back()]<m)
q.pop_back();
q.push_back(u);
while(l[u]-l[q.front()]>lim)
q.pop_front();
int v=q.front();
while(l[u]-l[h]>lim)h=nxt[h];
ans=min(ans,max(f[v],max(l[h],l[T]-l[u])));
}
printf("%d\n",ans);
}
int main(){
init();
diameter();
get();
solve();
}

欢迎dalaoD我,鄙视dalao装弱%我。。。

[bzoj1999][noip2007]Core树网的核的更多相关文章

  1. [BZOJ1999][codevs1167][Noip2007]Core树网的核

    [BZOJ1999][codevs1167][Noip2007]Core树网的核 试题描述 设T=(V, E, W) 是一个无圈且连通的无向图(也称为无根树),每条边带有正整数的权,我们称T为树网(t ...

  2. BZOJ_1999_[Noip2007]Core树网的核_单调队列+树形DP

    BZOJ_1999_[Noip2007]Core树网的核_单调队列+树形DP Description 设T=(V, E, W) 是一个无圈且连通的无向图(也称为无根树),每条边带有正整数的权,我们称T ...

  3. 【bzoj1999】[Noip2007]Core树网的核 树的直径+双指针法+单调队列

    题目描述 给出一棵树,定义一个点到一条路径的距离为这个点到这条路径上所有点的距离的最小值.求一条长度不超过s的路径,使得所有点到这条路径的距离的最大值最小. 输入 包含n行: 第1行,两个正整数n和s ...

  4. 洛谷 1099 ( bzoj 1999 ) [Noip2007]Core树网的核

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1999 <算法竞赛进阶指南>346页.https://www.cnblogs.co ...

  5. [Noip2007]Core树网的核

    嘟嘟嘟 首先求树的直径两次bfs即可,实际上bfs就是最短路,因为树上路径是唯一的,所以用任何一种遍历方法都行(spfa和dijkstra当然也可以). 可以证明,只要求出任意一条直径就行了,为什么呢 ...

  6. bzoj 1999: [Noip2007]Core树网的核【树的直径+单调队列】

    我要懒死了,所以依然是lyd的课件截图 注意是min{max(max(d[uk]),dis(u1,ui),dis(uj,un))},每次都从这三个的max里取min #include<iostr ...

  7. 【BZOJ1999】【NOIP2007】树网的核 单调队列优化DP

    题目描述 题目很长,大家自己去看吧. bzoj vijos 原题\(n\leq 300\) 加强版\(n\leq 500000\) 题解 这种东西当然要猜结论的啦,否则会比较麻烦. 结论1:如果有很多 ...

  8. 【noip2007】树网的核

    题解: 首先我们要知道一个性质:如果有多条直径 这个核不论在哪条直径上 答案都是一样的 这样我们就可以随便找一条直径 在这条直径上枚举核的位置 并且dfs预处理maxlon[i] (i在直径上) 表示 ...

  9. 【DFS好题】BZOJ1999- [Noip2007]Core树网的核(数据加强版)

    NOIP的数据好水,一开始有好几个错结果NOIP数据就水过了?? [题目大意] 求无根树的直径上一段不超过S长的链,使得偏心距最小.具体概念见原题. [思路] 首先明确几个性质: (1)对于树中的任意 ...

随机推荐

  1. Razor 语法糖常规用法

    1.隐匿代码表达式 例: @model.name 会将表达式的值计算并写入到响应中,输入时采用html编码方式 2.显示表达式 例:@(model.name)会将输入@model.name字符串 3. ...

  2. 启动selenium server

    java -jar selenium-server-standalone-2.37.0.jar

  3. API:Sign签名的执行流程

    Sign签名存在目的:为了防止不法分子修改参数数据,进而攻击服务器,导致数据泄露或从中获得利益    例如:一个接口是用户把积分转帐给他的朋友,修改后,变为转帐到攻击者的帐户,这样,攻击者就能得到利益 ...

  4. 前端必须要掌握的几个CSS3的属性

    随着Css3和html5的风靡,越来越多的前端人员开始学习Css3,今天的文章就是来说说前端应该掌握10个Css3属性. 1. Border-radius Border-radius是一大堆CSS3属 ...

  5. Prim算法求最大权,POJ(2485)

    题目链接:http://poj.org/problem?id=2485 解题报告: 这里有一点要注意的是,第一个点时,dis数组还没有初始化,还全部为inf.第一次来到更新权时,才把邻接矩阵的数据存到 ...

  6. Object类的常用方法

    Object类是Java中所有类的始祖.如果没有明确的指定继承,则默认继承Object类.在Java中除了基本类型外都是Object类型的对象,包括数组. 1)equals方法 Object: pub ...

  7. 移动页面请使用CSS3动画

    说到动画,我们一般会使用jQuery 中的animate(); 在PC浏览器中,是很方便的,由于PC的高性能,这种不断修改DOM的做法确实不会出现太大的问题,但是在手机端就不同了. 手机上使用jQue ...

  8. JS isArray、typeof、instanceof

    Array.isArray() 用来检验是不是数组 var a = [1,2,3] console.log(typeof a); // object console.log(Array.isArray ...

  9. SEOer必读:50个网站推广方法

    1.论坛推广 这里所说的论坛推广绝对不是在论坛里一个一个版贴广告,也不是将网站地址加在签名里然后疯狂刷屏,那样既耗费精力而且效果也不见得好,论坛管理员只要点几下鼠标就能将你的帖子全部删除,顺便封掉你的 ...

  10. BAT及各大互联网公司2014前端笔试面试题

    很多面试题是我自己面试BAT亲身经历碰到的.整理分享出来希望更多的前端er共同进步吧,不仅适用于求职者,对于巩固复习前端基础更是大有裨益. 而更多的题目是我一路以来收集的,也有往年的,答案不确保一定正 ...