题解 [NOIP2015]运输计划
题解 [NOIP2015]运输计划
题面
解析
首先肯定是要求出每条路径的长度.
这个用节点到根的前缀和就行了(一开始脑抽写了个线段树...)
然后有一个显然的类似贪心的想法,
就是你改造的边肯定在最长的路径上,
(不然没有*用)
那么考虑枚举最长的路径上的边,计算改造它的答案,
对于边\(x\),路径可以分为两类:经过它的和不经过它的.
在它被改造后,经过它的路径就都减少了它的长度,
于是最长的就还是这条最长的路径,
而没经过它的就没有受到影响,取最长的路径,
因此删掉\(x\)的答案就是上面两种情况的\(max\),
但关键就是怎么求第二种情况:没经过\(x\)的最长的路径.
枚举边再一个个求似乎不可行,
我们可以考虑枚举路径计算它对边的贡献,
显然没在这条路径上的边都可以被这条路径更新,
于是考虑树剖,
这样一条路径就被划分成了若干个区间,
而区间的补集就是它要更新的边,
把边的答案对应到深度较大的点上,用线段树维护最大值即可.
code(代码挺长但仔细康康应该能懂):
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define fot for
#define ls(a) a<<1
#define rs(a) a<<1|1
#define fre(x) freopen(x".in","r",stdin),freopen(x".out","w",stdout)
using namespace std;
inline int read(){
int sum=0,f=1;char ch=getchar();
while(ch>'9'||ch<'0'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
return f*sum;
}
const int N=3000005;
struct ques{int s,t,w;}q[N];
struct edge{int to,next,w;}e[N<<1];
struct tree{int l,r,sum,maxn,tag;}t[N<<1];
struct qujian{int l,r;}sta[N];
struct node{int size,son,id,fa,dep,top,val;}a[N];
int n,m,tot,pla[N],ans;
int head[N],cnt,s[N];
inline void add(int x,int y,int w){
e[++cnt]=(edge){head[x],y,w};head[x]=cnt;
}
inline void dfs1(int x,int fa){
a[x].size=1;a[x].fa=fa;
a[x].dep=a[fa].dep+1;
for(int i=head[x];i;i=e[i].to){
int k=e[i].next;
if(k==fa) continue;
s[k]=s[x]+e[i].w;
dfs1(k,x);a[k].val=e[i].w;
a[x].size+=a[k].size;
if(a[k].size>a[a[x].son].size) a[x].son=k;
}
}
inline void dfs2(int x,int top){
a[x].top=top;a[x].id=++tot;
pla[tot]=x;
if(a[x].son) dfs2(a[x].son,top);
for(int i=head[x];i;i=e[i].to){
int k=e[i].next;
if(k==a[x].fa||k==a[x].son) continue;
dfs2(k,k);
}
}
inline void build(int p,int l,int r){
t[p].l=l;t[p].r=r;
if(l==r) {t[p].sum=a[pla[l]].val;return ;}
int mid=(l+r)>>1;
build(ls(p),l,mid);build(rs(p),mid+1,r);
}
inline int lca(int x,int y){
while(a[x].top!=a[y].top){
if(a[a[x].top].dep<a[a[y].top].dep) swap(x,y);
x=a[a[x].top].fa;
}
if(a[x].dep>a[y].dep) swap(x,y);
return x;
}
inline int dis(int x,int y){
return s[x]+s[y]-2*s[lca(x,y)];
}
inline bool cmp(ques a,ques b){return a.w>b.w;}
inline void Pushup(int p){
t[p].maxn=max(t[ls(p)].maxn,t[rs(p)].maxn);
}
inline void Pushdown(int p){
int ret=t[p].tag;
t[ls(p)].tag=max(t[ls(p)].tag,ret);
t[rs(p)].tag=max(t[rs(p)].tag,ret);
t[ls(p)].maxn=max(t[ls(p)].maxn,ret);
t[rs(p)].maxn=max(t[rs(p)].maxn,ret);
}
inline void change(int p,int l,int r,int w){
if(l>r||!l||!r) return ;
if(t[p].l>=l&&t[p].r<=r){
t[p].maxn=max(t[p].maxn,w);
t[p].tag=max(t[p].tag,w);
return ;
}
Pushdown(p);
int mid=(t[p].l+t[p].r)>>1;
if(l<=mid) change(ls(p),l,r,w);
if(r>mid) change(rs(p),l,r,w);
Pushup(p);
}
inline bool cmp1(qujian a,qujian b){return a.l<b.l;}
inline void find(int x,int y,int w){
int top=0;
while(a[x].top!=a[y].top){
if(a[a[x].top].dep<a[a[y].top].dep) swap(x,y);
sta[++top]=(qujian){a[a[x].top].id,a[x].id};
x=a[a[x].top].fa;
}
if(a[x].dep>a[y].dep) swap(x,y);
sta[++top]=(qujian){a[x].id+1,a[y].id};
int L=1,R;
sort(sta+1,sta+top+1,cmp1);
for(int i=1;i<=top;i++){
R=sta[i].l-1;
change(1,L,R,w);
L=sta[i].r+1;
}
change(1,L,n,w);
}
inline int search(int p,int x){
if(t[p].l==t[p].r) return t[p].maxn;
Pushdown(p);
int mid=(t[p].l+t[p].r)>>1;
if(x<=mid) return search(ls(p),x);
else return search(rs(p),x);
}
inline void solve(int x,int y){
int pp=lca(x,y);
while(x!=pp){
ans=min(ans,max(q[1].w-a[x].val,search(1,a[x].id)));
x=a[x].fa;
}
while(y!=pp){
ans=min(ans,max(q[1].w-a[y].val,search(1,a[y].id)));
y=a[y].fa;
}
}
signed main(){
n=read();m=read();
for(int i=1;i<n;i++){
int x=read(),y=read(),w=read();
add(x,y,w);add(y,x,w);
}
for(int i=1;i<=m;i++){
q[i].s=read(),q[i].t=read();
}
dfs1(1,0);dfs2(1,1);
build(1,1,n);
for(int i=1;i<=m;i++){
q[i].w=dis(q[i].s,q[i].t);
}
for(int i=1;i<=m;i++) find(q[i].s,q[i].t,q[i].w);
sort(q+1,q+m+1,cmp);
ans=q[1].w;
fot(int i=1;i<=m;i++)
if(q[i].w==q[1].w) solve(q[i].s,q[i].t);
else break;
printf("%d\n",ans);
return 0;
}
题解 [NOIP2015]运输计划的更多相关文章
- NOIP2015 运输计划(bzoj4326)
4326: NOIP2015 运输计划 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 886 Solved: 574[Submit][Status] ...
- [BZOJ4326][codevs4632][codevs5440][UOJ#150][NOIP2015]运输计划
[BZOJ4326][codevs4632][codevs5440][UOJ#150][NOIP2015]运输计划 试题描述 公元 2044 年,人类进入了宇宙纪元. L 国有 n 个星球,还有 n− ...
- [NOIP2015]运输计划 D2 T3 LCA+二分答案+差分数组
[NOIP2015]运输计划 D2 T3 Description 公元2044年,人类进入了宇宙纪元. L国有n个星球,还有n-1条双向航道,每条航道建立在两个星球之间,这n-1条航道连通了L国的所有 ...
- bzoj 4326: NOIP2015 运输计划
4326: NOIP2015 运输计划 Time Limit: 30 Sec Memory Limit: 128 MB Description 公元 2044 年,人类进入了宇宙纪元.L 国有 n 个 ...
- NOIP2015 运输计划(二分+LCA+差分)
4326: NOIP2015 运输计划 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 308 Solved: 208[Submit][Status] ...
- 数据结构(树链剖分):COGS 2109. [NOIP2015] 运输计划
2109. [NOIP2015] 运输计划 ★★★ 输入文件:transport.in 输出文件:transport.out 简单对比时间限制:1 s 内存限制:256 MB [题目描 ...
- BZOJ 4326 NOIP2015 运输计划 (二分+树上差分)
4326: NOIP2015 运输计划 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 1930 Solved: 1231[Submit][Statu ...
- cogs2109 [NOIP2015] 运输计划
cogs2109 [NOIP2015] 运输计划 二分答案+树上差分. STO链剖巨佬们我不会(太虚伪了吧 首先二分一个答案,下界为0,上界为max{路径长度}. 然后判断一个答案是否可行,这里用到树 ...
- LOJ2425 NOIP2015 运输计划 【二分+LCA+树上差分】*
LOJ2425 NOIP2015 运输计划 LINK 题意:给你一颗树,可以将任意一条边的权值变成0,然后求m条路径的长度的最小值 思路: 先二分最后的距离ans,然后我们把路程大于ans的所有路径拿 ...
随机推荐
- WUSTOJ 1275: 男神的逆袭(Java)
1275: 男神的逆袭 题目 计算两个日期相差的天数.更多内容点击标题. 分析 下面说一下我的思路(自己写的,无扩展性): 给定一个日期,首先计算这个日期是这一年的第多少天. 给定两个日期,直 ...
- Scratch编程:牛顿的苹果——地心引力
牛顿的苹果 同学们,你们知道牛顿的苹果的故事吗? 传说1665年秋季,牛顿坐在自家院中的苹果树下苦思着行星绕日运动的原因.这时,一只苹果恰巧落下来,它落在牛顿的脚边.就是这个偶尔的瞬间,牛顿发现了苹果 ...
- k8s-部署策略
在Kubernetes中有几种不同的方式发布应用,所以为了让应用在升级期间依然平稳提供服务,选择一个正确的发布策略就非常重要了. 选择正确的部署策略是要依赖于我们的业务需求的,下面我们列出了一些可能会 ...
- hdu 6377 度度熊看球赛 (dp)
大意: $n$对情侣, $2n$个座位, 对于一个方案, 若$k$对情侣相邻, 则喧闹值增加$D^k$, 求喧闹值期望. 跟CF 840C一样, 设$dp[i][j]$为$i$个人, 有$j$对情侣相 ...
- Luogu4770 NOI2018你的名字(后缀自动机+线段树合并)
先考虑l=1,r=n,并且不要求本质不同的情况.对原串建SAM,将询问串在上面跑,得到每个前缀的最长匹配后缀即可得到答案. 然后考虑本质不同.对询问串也建SAM,统计每个节点的贡献,得到该点right ...
- 轻松搭建CAS 5.x系列(8)-在CAS Server增加双因素认证(DUO版)
概述说明 为了让系统更加安全,很多登录会加入双因素认证.何为双因素,如果把登陆作为开一扇门的话,那就是在原来的锁上再加一把锁,第二锁用新的钥匙,这样安全系数就更加高了. CAS是通过账号名和密码来认证 ...
- python 练习:函数1
习题: 定义一个方法 func,该func可以引入任意多的整型参数,结果返回其中最大与最小的值. def func(**args): return max(args),min(args) 定义一个方法 ...
- 一篇文章彻底搞懂异步,同步,setTimeout,Promise,async
之前翻看别的大佬的博客看到了关于setTimeout,promise还有async执行顺序的文章.观看了几篇之后还是没有怎么看懂,于是自己开始分析代码,并整理了此文章,我相信通过此文章朋友们能对异步同 ...
- OSI的七层网络模型
一.概念七层模型,亦称OSI(Open System Interconnection)参考模型,是参考模型是国际标准化组织(ISO)制定的一个用于计算机或通信系统间互联的标准体系.它是一个七层的.抽象 ...
- 利用PL/SQL从Oracle数据库导出和导入数据
转自:https://www.jb51.net/article/109768.htm 本文实例为大家分享了使用PL/SQL从Oracle数据库导出和导入数据的方法,供大家参考,具体内容如下 1.导出数 ...