Codeforces Round #520 (Div. 2) E. Company(dfs序判断v是否在u的子树里+lca+线段树)
https://codeforces.com/contest/1062/problem/E
题意
给一颗树n,然后q个询问,询问编号l~r的点,假设可以删除一个点,使得他们的最近公共祖先深度最大。每次询问,输出删除的点和祖先的深度
思路
- 考虑dfs序来判断v是否在u的子树里:
dfn[u]<dfn[v]<=max(dfn[u的子树])
- 那么进一步拓展,dfs序之差越大,点就在越分离的地方,这些点的lca一定是lca(max(dfn[u]),min(dfn[u]))
- 这不仅知道了需要去掉哪个点(上述两个点之一),而且知道了怎么求点的lca
- 然后枚举去掉两个点求lca(lca(l,u-1),lca(u+1,r))
- 线段树维护最小,最大dfn的点
#include<bits/stdc++.h>
#define M 100005
#define pb push_back
#define inf 100000000
using namespace std;
struct N{
int l,r;
array<int,2>mi,ma;
}tr[M<<2];
vector<int>g[M];
int n,q,i,d[M],pre[M],u,v,l,r,a,b,dfn,f[M][30];
void push_up(int o){
tr[o].mi=min(tr[o<<1].mi,tr[o<<1|1].mi);
tr[o].ma=max(tr[o<<1].ma,tr[o<<1|1].ma);
}
int lca(int u,int v){
if(d[v]>d[u])swap(u,v);
int dis=d[u]-d[v];
for(int i=20;i>=0;i--)if(dis&(1<<i))u=f[u][i];
if(u==v)return u;
for(int i=20;i>=0;i--)
if(f[u][i]!=f[v][i]){u=f[u][i];v=f[v][i];}
return f[u][0];
}
void dfs(int u,int fa){
f[u][0]=fa;for(int i=1;i<=20;i++)f[u][i]=f[f[u][i-1]][i-1];
d[u]=d[fa]+1;pre[u]=++dfn;
//cout<<u<<" "<<d[u]<<endl;
for(int i=0;i<g[u].size();i++){
int v=g[u][i];
dfs(v,u);
}
}
void build(int o,int l,int r){
tr[o].l=l;tr[o].r=r;
if(l==r){
tr[o].ma=tr[o].mi={pre[l],l};
return;
}
int mid=(l+r)/2;
build(o<<1,l,mid);
build(o<<1|1,mid+1,r);
push_up(o);
}
array<int,2> qy1(int o,int L,int R){
int l=tr[o].l,r=tr[o].r,mid=(l+r)/2;
if(L<=l&&r<=R)return tr[o].mi;
array<int,2>ans={inf,n+1};
if(L<=mid)ans=min(ans,qy1(o<<1,L,R));
if(R>mid)ans=min(ans,qy1(o<<1|1,L,R));
return ans;
}
array<int,2> qy2(int o,int L,int R){
int l=tr[o].l,r=tr[o].r,mid=(l+r)/2;
if(L<=l&&r<=R)return tr[o].ma;
array<int,2>ans={0,n+1};
if(L<=mid)ans=max(ans,qy2(o<<1,L,R));
if(R>mid)ans=max(ans,qy2(o<<1|1,L,R));
return ans;
}
int main(){
dfn=0;
cin>>n>>q;
for(i=2;i<=n;i++){scanf("%d",&u);g[u].pb(i);}
d[0]=-1; dfs(1,0);
build(1,1,n);
while(q--){
scanf("%d%d",&l,&r);
u=qy1(1,l,r)[1];v=qy2(1,l,r)[1];
if(u==l)a=lca(qy1(1,l+1,r)[1],qy2(1,l+1,r)[1]);
else if(u==r)a=lca(qy1(1,l,r-1)[1],qy2(1,l,r-1)[1]);
else a=lca(lca(qy1(1,l,u-1)[1],qy2(1,l,u-1)[1]),lca(qy1(1,u+1,r)[1],qy2(1,u+1,r)[1]));
if(v==l)b=lca(qy1(1,l+1,r)[1],qy2(1,l+1,r)[1]);
else if(v==r)b=lca(qy1(1,l,r-1)[1],qy2(1,l,r-1)[1]);
else b=lca(lca(qy1(1,l,v-1)[1],qy2(1,l,v-1)[1]),lca(qy1(1,v+1,r)[1],qy2(1,v+1,r)[1]));
//cout<<u<<" "<<a<<endl;
//cout<<v<<" "<<b<<endl;
if(d[a]>d[b])printf("%d %d\n",u,d[a]);
else printf("%d %d\n",v,d[b]);
}
}
Codeforces Round #520 (Div. 2) E. Company(dfs序判断v是否在u的子树里+lca+线段树)的更多相关文章
- Codeforces Round #520 (Div. 2)
Codeforces Round #520 (Div. 2) https://codeforces.com/contest/1062 A #include<bits/stdc++.h> u ...
- Codeforces Round #520 (Div. 2) Solution
A. A Prank Solved. 题意: 给出一串数字,每个数字的范围是$[1, 1000]$,并且这个序列是递增的,求最多擦除掉多少个数字,使得别人一看就知道缺的数字是什么. 思路: 显然,如果 ...
- Codeforces Round #459 (Div. 2) D. MADMAX DFS+博弈
D. MADMAX time limit per test 1 second memory limit per test 256 megabytes input standard input outp ...
- Codeforces Round #222 (Div. 1) A. Maze dfs
A. Maze 题目连接: http://codeforces.com/contest/377/problem/A Description Pavel loves grid mazes. A grid ...
- Codeforces Round #245 (Div. 2) C. Xor-tree DFS
C. Xor-tree Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/430/problem/C ...
- Codeforces Round #398 (Div. 2) C. Garland —— DFS
题目链接:http://codeforces.com/contest/767/problem/C 题解:类似于提着一串葡萄,用剪刀剪两条藤,葡萄分成了三串.问怎样剪才能使三串葡萄的质量相等. 首先要做 ...
- Codeforces Round #321 (Div. 2)C(tree dfs)
题意:给出一棵树,共有n个节点,其中根节点是Kefa的家,叶子是restaurant,a[i]....a[n]表示i节点是否有猫,问:Kefa要去restaurant并且不能连续经过m个有猫的节点有多 ...
- Codeforces Round #267 (Div. 2)D(DFS+单词hash+简单DP)
D. Fedor and Essay time limit per test 2 seconds memory limit per test 256 megabytes input standard ...
- Codeforces Round #527 (Div. 3)F(DFS,DP)
#include<bits/stdc++.h>using namespace std;const int N=200005;int n,A[N];long long Mx,tot,S[N] ...
随机推荐
- Codeforces Beta Round #57 (Div. 2)
Codeforces Beta Round #57 (Div. 2) http://codeforces.com/contest/61 A #include<bits/stdc++.h> ...
- 微信小程序template模板与component组件的区别和使用
前言: 除了component,微信小程序中还有另一种组件化你的方式template模板,这两者之间的区别是,template主要是展示,方法则需要在调用的页面中定义.而component组件则有自己 ...
- [剑指Offer]52-两个链表的第一个公共节点
题目链接 https://www.nowcoder.com/practice/6ab1d9a29e88450685099d45c9e31e46?tpId=13&tqId=11189&t ...
- scrapy的调试和环境安装技巧
1,先在settings中把ROBOTSTXT_OBEY = False 在主目录下面新建main文件 __autor__ = 'zhouli' __date__ = '2018/11/3 22:39 ...
- 再遇ibatisNet
11年在Mr刘的带领下第一次接触ibatisnet ,当时Mr刘很详细的很讲了xml里的写法还有配置文件之类的,但是随着时间越来越久远.很多东西都开始淡忘了. 如今,再次和它相遇,依然觉得很亲切,虽然 ...
- stark组件开发之提取公共视图函数
路由问题, 已经解决! 然后就是视图函数的问题了: 不想重复写的解决途径就是, python 类的继承了! 写一个基类, 基类定义 增删改查. 然后其他的,全部去继承他! from django. ...
- 使用JFinal实现使用MVC获取表单中的数据并将提示信息返回给另一jsp页面。
1.包结构 2.我们需要对web.xml进行配置: <?xml version="1.0" encoding="UTF-8"?> <web-a ...
- layer.confirm在ASP.NET控件onclick上面的应用方法
有些时候,你可能要修改控件的事件,元素本身.等,这个时候如何操作呢?下面提供一个思路: <asp:LinkButton Visible="false" ID="sh ...
- PAT 1023 组个最小数 (20)(代码+思路)
1023 组个最小数 (20)(20 分) 给定数字0-9各若干个.你可以以任意顺序排列这些数字,但必须全部使用.目标是使得最后得到的数尽可能小(注意0不能做首位).例如:给定两个0,两个1,三个5, ...
- java中 this 关键字的三种用法
Java中this的三种用法 调用属性 (1)this可以调用本类中的任何成员变量 调用方法(可省略) (2)this调用本类中的成员方法(在main方法里面没有办法通过this调用) 调用构造方法 ...