• 题目链接:

    https://www.luogu.org/problemnew/show/P1099

    https://www.lydsy.com/JudgeOnline/problem.php?id=1999 (加强版)

  • 分析:

    首先你需要\(O(N)\)求树的直径的前置技能,其实很简单,先随便找个根找到树上距离它最远的顶点\(S\),然后以\(S\)为根找树上距离\(S\)最远的顶点\(E\),\(S,E\)之间的路径就是树的直径

    1. 暴力枚举

      按照题目要求说的去做就好了,求出树的直径,然后根据贪心找长度小于等于s的路径搜一遍就好了,时间复杂度\(O(N^2)\) 代码难度小 可过NOIP数据

    2. 预处理扫描

      还是先求出树的直径,然后通过仔细分析偏心距怎么求,发现无非三种情况

      1.直径最左端顶点到路径最左端顶点距离

      2.路径上各点不经过直径上其他点到达的最大距离

      3.直径上最右端顶点到路径最右端顶点的距离

      我相信2理解不难,现在简单证明1情况(3同理)的正确性:假设路径最左端有一条向左延伸的路径其终点\(E'\)不是直径左端点而距离却更大,则违反直径定义,因为这样\(E'\)与路径左端点连接能形成一条更长的直径

      于是我们需要四遍DFS(也许你写的好的话并不需要这么多遍)

      前两遍就是求树的直径,用一个\(dmet[]\)将直径各点按顺序记录方便区间处理,\(diameter\)为直径长度

      第三遍求直径上各点不经过直径上其他点达到的最大距离,用\(d[]\)记录

      第四遍求直径左右段点到直径各点距离,用\(ld[],rd[]\)记录

      然后我们扫描长度小于等于s的路径(可能是一个点),找上述三种情况的最大值,找第二种情况最大值你可以用滑动窗口,ST Table,线段树在区间查找最大值,这里采用线段树

      时间复杂度:\(O(N\log N)\)

      可以过比较大的数据,这是加强版

      https://www.lydsy.com/JudgeOnline/problem.php?id=1999

  • 代码

    写得比较丑,见谅

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <algorithm>
#include <cctype>
#include <cstring>
#include <queue>
#define ll long long
#define ri register int
using namespace std;
const int maxn=500005;
const int maxm=1000005;
const int inf=0x7fffffff;
template <class T>inline void read(T &x){
x=0;int ne=0;char c;
while(!isdigit(c=getchar()))ne=c=='-';
x=c-48;
while(isdigit(c=getchar()))x=(x<<3)+(x<<1)+c-48;
x=ne?-x:x;
return ;
}
struct Edge{
int ne,to,dis;
}edge[maxm];
int h[maxn],num_edge=0,mx=-inf;
bool vis[maxn];
void add(int f,int to,int dis){
edge[++num_edge].ne=h[f];
edge[num_edge].to=to;
edge[num_edge].dis=dis;
h[f]=num_edge;
}
int n,s,head,tail;
void dfs_1(int fa,int cur,int cnt){
for(ri i=h[cur];i;i=edge[i].ne){
if(edge[i].to!=fa)dfs_1(cur,edge[i].to,cnt+edge[i].dis);
}
if(cnt>mx){
mx=cnt,head=cur;
}
return ;
}
int pre[maxn],dmet[maxn],d[maxn],diameter;
void dfs_2(int fa,int cur,int cnt){
for(ri i=h[cur];i;i=edge[i].ne){
if(edge[i].to!=fa){
pre[edge[i].to]=cur;
dfs_2(cur,edge[i].to,cnt+edge[i].dis);
}
}
if(cnt>mx){
mx=cnt,tail=cur;
}
return ;
}
int root;
void dfs_3(int fa,int cur,int cnt){
for(ri i=h[cur];i;i=edge[i].ne){
int v=edge[i].to;
if(!vis[v]&&v!=fa){
dfs_3(cur,v,cnt+edge[i].dis);
}
}
if(cnt>mx){
mx=cnt,d[root]=cnt;
}
return ;
}
int ld[maxn],rd[maxn],fun[maxn];
void dfs_4(int fa,int cur,int cnt){
for(ri i=h[cur];i;i=edge[i].ne){
int v=edge[i].to;
if(vis[v]&&v!=fa){
dfs_4(cur,v,cnt+edge[i].dis);
}
}
ld[fun[cur]]=cnt,rd[fun[cur]]=diameter-cnt;
return ;
}
int maxx[maxn<<2],L,R;
void build(int now,int l,int r){
if(l==r){
maxx[now]=d[dmet[l]];
return;
}
int mid=(l+r)>>1;
build(now<<1,l,mid);
build(now<<1|1,mid+1,r);
maxx[now]=max(maxx[now<<1],maxx[now<<1|1]);
return ;
}
int query(int now,int l,int r){
if(L<=l&&r<=R){
return maxx[now];
}
int mid=(l+r)>>1,ans=-inf;
if(L<=mid)ans=max(ans,query(now<<1,l,mid));
if(mid<R)ans=max(ans,query(now<<1|1,mid+1,r));
return ans;
}
int main(){
int x,y,z;
read(n),read(s);
for(ri i=1;i<n;i++){
read(x),read(y),read(z);
add(x,y,z);
add(y,x,z);
}
mx=-inf;
dfs_1(0,1,0);
mx=-inf;
dfs_2(0,head,0);
diameter=mx; //两次DFS找直径
/*------------------*/
int tmp=tail,tot=0;
while(tmp!=head){
dmet[++tot]=tmp;
fun[tmp]=tot;
vis[tmp]=1;
tmp=pre[tmp];
}
dmet[++tot]=head;
fun[head]=tot;
vis[head]=1; //记录直径
/*------------------*/
for(ri i=1;i<=tot;i++){
root=dmet[i];
mx=-inf;
dfs_3(0,dmet[i],0);
} //找从直径各点不经过其他直径上的点走过的最大长度
dfs_4(0,tail,0);//找直径两端点到直径各点距离
build(1,1,tot);//建线段树,不会ST Table的我
/*-----------------*/
int ans=inf,l=1,r=1;
while(ld[r+1]-ld[l]<=s&&r!=tot)r++;
//cout<<tot<<'*'<<endl;
while(r<=tot){//cout<<l<<' '<<r<<endl;
L=l,R=r;
ans=min(ans,max(query(1,1,tot),max(ld[l],rd[r])));
if(r==tot)break;
l++;
while(ld[r+1]-ld[l]<=s&&r!=tot)r++;
}
printf("%d\n",ans);
return 0;
}

luogu题解 P1099 【树网的核】树的直径变式+数据结构维护的更多相关文章

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

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

  2. luogu题解 P3629 【[APIO2010]巡逻】树的直径变式

    题目链接: https://www.luogu.org/problemnew/show/P3629 分析 最近被众多dalao暴虐,这道题傻逼地调了两天才知道错哪 不过这题比较良心给你一个容易发现性质 ...

  3. 洛谷 P1099 树网的核

    P1099 树网的核 题目描述 设T=(V, E, W) 是一个无圈且连通的无向图(也称为无根树),每条边到有正整数的权,我们称T为树网(treebetwork),其中V,E分别表示结点与边的集合,W ...

  4. bzoj1999 / P1099 树网的核

    P1099 树网的核 (bzoj数据加强) 前置知识:树的直径 (并不想贴我的智障写法虽然快1倍但内存占用极大甚至在bzoj上MLE) 正常写法之一:用常规方法找到树的直径,在直径上用尺取法找一遍,再 ...

  5. P1099 树网的核——模拟+树形结构

    P1099 树网的核 无根树,在直径上找到一条长度不超过s的路径,使得最远的点距离这条路径的距离最短: 首先两遍dfs找到直径(第二次找的时候一定要吧father[]清零) 在找到的直径下枚举长度不超 ...

  6. P2491 消防/P1099 树网的核

    P2491 消防/P1099 树网的核 双倍经验,双倍快乐. 题意 在一个树上选择一段总长度不超过\(s\)的链使所有点到该链距离的最大值最小. 输出这个最小的值. 做法 Define:以下\(s\) ...

  7. #P1099 树网的核 题解

    题目描述 pdf 题解 这一题,刚开始看题目感觉好像很难,题目又长……一看数据范围,呵呵. 已经给出来这是个DAG,所以不用担心连通性的问题.那么怎么做呢? 朴素的做法是把树的直径的两个端点都统计出来 ...

  8. [NOIP2007] 提高组 洛谷P1099 树网的核

    题目描述 设T=(V, E, W) 是一个无圈且连通的无向图(也称为无根树),每条边到有正整数的权,我们称T为树网(treebetwork),其中V,E分别表示结点与边的集合,W表示各边长度的集合,并 ...

  9. 树网的核[树 floyd]

    描述 设T=(V, E, W) 是一个无圈且连通的无向图(也称为无根树),每条边到有正整数的权,我们称T为树网(treebetwork),其中V,E分别表示结点与边的集合,W表示各边长度的集合,并设T ...

随机推荐

  1. Java-AESUtil

    在线版 http://tool.chacuo.net/cryptaes 要使用 AES/CBC/PKCS7Padding 模式需要添加依赖 <!--AES/CBC/PKCS7Padding--& ...

  2. git *** Please tell me who you are.错误

    GIT 中提示 please tell me who you are   如果使用git过程中出现了,please tell me who you are ,需要设置一下使用者的身份. 1.git c ...

  3. leetcode-hard-array-41. First Missing Positive-NO

    mycode class Solution(object): def firstMissingPositive(self, nums): """ :type nums: ...

  4. 判断List是否为空的问题

    今天公司安排给页面调试Bug,感觉公司人员写的判断判断List是否为空存在一定的问题,公司判断是list!=null这是完全不对的,这只会判断是否有list对象.如果为空,他也会执行if(list!= ...

  5. 导出Excel/Pdf/txt/json/XML/PNG/CSV/SQL/MS-Word/ Ms-Powerpoint/等通过tableExport.js插件来实现

    首先去我的云盘下载需要的js: 链接:https://pan.baidu.com/s/13vC-u92ulpx3RbljsuadWw 提取码:mo8m 页面代码: <!DOCTYPE html& ...

  6. OGG 从Oracle备库同步数据至kafka

    OGG 从Oracle备库同步数据至kafka Table of Contents 1. 目的 2. 环境及规划 3. 安装配置JDK 3.1. 安装jdk 3.2. 配置环境变量 4. 安装Data ...

  7. 阶段3 3.SpringMVC·_06.异常处理及拦截器_1 SpringMVC异常处理之分析和搭建环境

    异常一级一级的抛出 前端控制器,调用异常处理器组件 搭建环境 注意下面两个的结尾的名称要个 Module Name对应起来. 导入开发的坐标 复制upload这个项目里面的 编程和生成 改成1.8 配 ...

  8. 初探ASP.NET Web API (转)

    http://www.cnblogs.com/mejoy/p/6402821.html 什么是ASP.NET Web API? 官方的解释是 ASP.NET Web API is a framewor ...

  9. 第一个简单APP设计图

    以下是我画出来的最简单的手机UI设计图哟,以后慢慢积累吧.... 其实使用很简单,很多控件都有,直接使用就好....还是多动手吧,相信自己之后能很好的掌握这个的使用哟!!!!!!

  10. Bug解决:mysql 创建表字段Double类型长度

    excel导入数据进行新增时,发现安装高度和可视距离在数据库创建都是double类型 程序跑完,执行成功后,数据库的数据是2,小数点后的数据没有了 打印sql并执行后发现sql并没有错误, 检查数据库 ...