【CF725G】Messages on a Tree 树链剖分+线段树
【CF725G】Messages on a Tree
题意:给你一棵n+1个节点的树,0号节点是树根,在编号为1到n的节点上各有一只跳蚤,0号节点是跳蚤国王。现在一些跳蚤要给跳蚤国王发信息。具体的信息传输过程如下:
1.信息的发起者把信息上传给他父亲节点处的跳蚤,然后自身进入等待状态。
3.跳蚤国王在收到信息时会将信息立刻下传到发来信息的那个儿子,跳蚤国王可以在同一时刻下传多份信息。
4.上传:a把信息传给b。如果b正处于等待状态,则b会立刻将a发来的信息下传回去。如果同时有好多个信息传给b,则b会接受其中发起者编号最小的那个,并将其余的下传回去。如果上传成功,b会接着把信息上传给它的父亲,然后自身进入等待状态。
5.下传:b把信息传给a。解除a的等待状态,接着a继续把信息向着发起者下传过去。如果同时有信息上传和下传到a,视为先下传再上传。
6.当发起者收到下传回来的信息后,传输结束。
7.上传和下传所需时间都是1秒。
8.如果信息的发起者在发出信息时正处于等待状态,该信息直接结束。
现在给你m个时间,第i个事件形如:编号为a的跳蚤在t时刻发起了一次信息传输。现在问你每个信息传输的结束时间是多少。
题解:先将所有时间按$dep_a+t$排序,不难发现排序后只有前面的事件会影响后面的事件,然后我们依次处理每个事件。如果a在上传过程中停在了节点v处,而v正在等待来自b的消息,则一定满足:$dep_a-dep_v+t_a<T_b-(dep_b-dep_v)$,其中T表示结束时间。所以我们可以用树链剖分+线段树维护,然后在链上二分即可。
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define lson x<<1
#define rson x<<1|1
using namespace std;
const int maxn=200010;
const int inf=2000000000;
int n,m,cnt;
int to[maxn],nxt[maxn],head[maxn],fa[maxn],p[maxn],q[maxn],dep[maxn],siz[maxn],son[maxn],top[maxn];
int sd[maxn<<2],s[maxn<<2],tag[maxn<<2],ans[maxn];
struct node
{
int x,t,org;
}t[maxn];
void dfs(int x,int tp)
{
top[x]=tp;
p[x]=++q[0],q[q[0]]=x;
if(son[x]) dfs(son[x],tp);
for(int i=head[x];i!=-1;i=nxt[i]) if(to[i]!=son[x]) dfs(to[i],to[i]);
}
inline void add(int a,int b)
{
to[cnt]=b,nxt[cnt]=head[a],head[a]=cnt++;
}
void build(int l,int r,int x)
{
s[x]=tag[x]=-inf;
if(l==r)
{
sd[x]=dep[q[l]]<<1;
return ;
}
int mid=(l+r)>>1;
build(l,mid,lson),build(mid+1,r,rson);
sd[x]=max(sd[lson],sd[rson]);
}
inline void upd(int x,int y)
{
s[x]=max(s[x],sd[x]+y),tag[x]=max(tag[x],y);
}
inline void pushdown(int x)
{
if(tag[x]!=-inf) upd(lson,tag[x]),upd(rson,tag[x]),tag[x]=-inf;
}
void updata(int l,int r,int x,int a,int b,int c)
{
if(a<=l&&r<=b)
{
upd(x,c);
return ;
}
pushdown(x);
int mid=(l+r)>>1;
if(a<=mid) updata(l,mid,lson,a,b,c);
if(b>mid) updata(mid+1,r,rson,a,b,c);
s[x]=max(s[lson],s[rson]);
}
int query(int l,int r,int x,int a,int b)
{
if(a<=l&&r<=b) return s[x];
pushdown(x);
int mid=(l+r)>>1;
if(b<=mid) return query(l,mid,lson,a,b);
if(a>mid) return query(mid+1,r,rson,a,b);
return max(query(l,mid,lson,a,b),query(mid+1,r,rson,a,b));
}
bool cmp(const node &a,const node &b)
{
return (dep[a.x]+a.t==dep[b.x]+b.t)?(a.x<b.x):(dep[a.x]+a.t<dep[b.x]+b.t);
}
int solve(int x,int t)
{
if(x==11)
{
x++,x--;
}
int u=x;
while(u)
{
if(query(1,n,1,p[top[u]],p[u])<=dep[x]+t)
{
u=fa[top[u]];
continue;
}
int l=p[top[u]]+1,r=p[u]+1,mid;
while(l<r)
{
mid=(l+r)>>1;
if(query(1,n,1,mid,p[u])>dep[x]+t) l=mid+1;
else r=mid;
}
u=q[l-1];
break;
}
if(!u) u=1;
int ret=2*(dep[x]-dep[u])+t,v=x;
while(top[v]!=top[u])
{
updata(1,n,1,p[top[v]],p[v],ret-dep[x]),v=fa[top[v]];
}
updata(1,n,1,p[u],p[v],ret-dep[x]);
return ret;
}
inline int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<'0'||gc>'9') {if(gc=='-') f=-f; gc=getchar();}
while(gc>='0'&&gc<='9') ret=ret*10+(gc^'0'),gc=getchar();
return ret*f;
}
int main()
{
int i;
n=rd()+1,m=rd();
dep[1]=1;
memset(head,-1,sizeof(head));
for(i=2;i<=n;i++) fa[i]=rd()+1,add(fa[i],i),dep[i]=dep[fa[i]]+1;
for(i=n;i>=2;i--)
{
siz[i]++,siz[fa[i]]+=siz[i];
if(siz[i]>siz[son[fa[i]]]) son[fa[i]]=i;
}
dfs(1,1);
build(1,n,1);
for(i=1;i<=m;i++) t[i].x=rd()+1,t[i].t=rd(),t[i].org=i;
sort(t+1,t+m+1,cmp);
for(i=1;i<=m;i++)
{
ans[t[i].org]=solve(t[i].x,t[i].t);
}
for(i=1;i<=m;i++) printf("%d ",ans[i]);
return 0;
}//10 9 0 0 0 3 0 4 5 3 8 6 1 6 1 13 5 14 1 16 5 18 7 18 10 21 6 22 4 22
【CF725G】Messages on a Tree 树链剖分+线段树的更多相关文章
- Aizu 2450 Do use segment tree 树链剖分+线段树
Do use segment tree Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.bnuoj.com/v3/problem_show ...
- 【POJ3237】Tree(树链剖分+线段树)
Description You are given a tree with N nodes. The tree’s nodes are numbered 1 through N and its edg ...
- POJ3237 Tree 树链剖分 线段树
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - POJ3237 题意概括 Description 给你由N个结点组成的树.树的节点被编号为1到N,边被编号为1 ...
- Spoj Query on a tree SPOJ - QTREE(树链剖分+线段树)
You are given a tree (an acyclic undirected connected graph) with N nodes, and edges numbered 1, 2, ...
- Water Tree CodeForces 343D 树链剖分+线段树
Water Tree CodeForces 343D 树链剖分+线段树 题意 给定一棵n个n-1条边的树,起初所有节点权值为0. 然后m个操作, 1 x:把x为根的子树的点的权值修改为1: 2 x:把 ...
- 【BZOJ-2325】道馆之战 树链剖分 + 线段树
2325: [ZJOI2011]道馆之战 Time Limit: 40 Sec Memory Limit: 256 MBSubmit: 1153 Solved: 421[Submit][Statu ...
- POJ3237 (树链剖分+线段树)
Problem Tree (POJ3237) 题目大意 给定一颗树,有边权. 要求支持三种操作: 操作一:更改某条边的权值. 操作二:将某条路径上的边权取反. 操作三:询问某条路径上的最大权值. 解题 ...
- B20J_3231_[SDOI2014]旅行_树链剖分+线段树
B20J_3231_[SDOI2014]旅行_树链剖分+线段树 题意: S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,城市信仰不同的宗教,为了方便,我们用不同的正整数代表各种宗教. S国 ...
- BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 )
BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 ) 题意分析 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 ...
随机推荐
- UNIX环境编程学习笔记(15)——进程管理之进程终止
lienhua342014-10-02 1 进程的终止方式 进程的终止方式有 8 种,其中 5 种为正常终止,它们是 1. 从 main 返回. 2. 调用 exit. 3. 调用_exit 或_Ex ...
- unity之UGUI屏幕分辨率調整
1.Canvas的屬性配置: 2.Canvas Scaler的屬性配置: 3.根據不同的屏幕比例動態寫改碩放基準: public float standard_width = 800f; //初始宽度 ...
- 2018-10-29 A股主要指数的市盈率(PE)估值高度
全指材料(SH000987) - 2018-10-29日,当前值:11.9289,平均值:30.66,中位数:26.1407,当前 接近历史新低.全指材料(SH000987)的历史市盈率PE详情 全指 ...
- HTML5 标准规范完成了
万维网联盟(W3C)昨天宣布,HTML5 标准规范终于最终制定完成了,并已公开发布.对于前端工程师来说,这无疑是一个振奋人心的好消息. 众所周知,HTML5改变了互联网,将成为 ...
- Android刮刮卡效果
不多说,直接上代码: package com.example.test; import android.app.Activity; import androi ...
- Maven 多项目依赖,需要验证artifact的output root中是否包含其他项目输出
- java中类相关注意事项
下面default类就是默认修饰符的类 1.Java中调用类中属性或方法(不管是否静态属性或方法)都要在类的方法中调用,虽然这个太基础,但今天想在类中调用静态类的静态变量,不能调用: 2.Java调用 ...
- CopyTransform
// TransformCopier.cs v 1.1 // homepage: http://wiki.unity3d.com/index.php/CopyTransform using Unity ...
- Streaming 101
开宗明义!本文根据Google Beam大神Tyler Akidau的系列文章<The world beyond batch: Streaming 101>(批处理之外的流式世界)整理而成 ...
- PHP错误 。Parse error: syntax error, unexpected T_INLINE_HTML, expecting T_ENDSWITCH or T_CASE or T_DEFAULT
If you wan't to use the alternative syntax for switch statements this won't work: <div> <?p ...