BZOJ_3626_[LNOI2014]LCA_离线+树剖

题意:

给出一个n个节点的有根树(编号为0到n-1,根节点为0)。一个点的深度定义为这个节点到根的距离+1。
设dep[i]表示点i的深度,LCA(i,j)表示i与j的最近公共祖先。
有q次询问,每次询问给出l r z,求$\sum\limits_{l\le i\le r}dep(LCA(i,z))$。
(即,求在[l,r]区间内的每个节点i与z的最近公共祖先的深度之和)

分析:

求$\sum\limits_{l\le i\le r}dep(LCA(i,z))$相当于把[l,r]区间内所有点依次插入。

每次把点到1路径上所有点+1,可以发现z到1路径上的点权和即为所求。

并且该操作具有区间可减性。那么我们把所有询问拆成[0~l-1],[0~r]两部分。

排个序离线计算。

代码:

#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
#define N 100050
#define LL long long
#define ls p<<1
#define rs p<<1|1
int head[N],to[N<<1],nxt[N<<1],cnt,n,m,r,mod=201314;
int t[N<<2],lz[N<<2],tot;
int siz[N],top[N],son[N],fa[N],dep[N],idx[N];
LL now,ans[N];
struct A{
int pos,z,flg,id;
}a[N];
bool cmp(const A &x,const A &y){
return x.pos<y.pos;
}
inline void add(int u,int v){
to[++cnt]=v;nxt[cnt]=head[u];head[u]=cnt;
}
void dfs1(int x,int y){
fa[x]=y;dep[x]=dep[y]+1;siz[x]=1;
for(int i=head[x];i;i=nxt[i]){
if(to[i]!=y){
dfs1(to[i],x);
siz[x]+=siz[to[i]];
if(siz[to[i]]>siz[son[x]])son[x]=to[i];
}
}
}
void dfs2(int x,int t){
top[x]=t;idx[x]=++tot;
if(son[x])dfs2(son[x],t);
for(int i=head[x];i;i=nxt[i]){
if(to[i]!=fa[x]&&to[i]!=son[x])dfs2(to[i],to[i]);
}
}
void pud(int l,int r,int p){
if(!lz[p])return ;
int mid=l+r>>1;
lz[ls]=(lz[ls]+lz[p])%mod;
lz[rs]=(lz[rs]+lz[p])%mod;
t[ls]=(t[ls]+lz[p]*(mid-l+1))%mod;
t[rs]=(t[rs]+lz[p]*(r-mid))%mod;
lz[p]=0;
}
void up(int l,int r,int x,int y,int c,int p){
if(x<=l&&r<=y){
lz[p]=(lz[p]+c)%mod;
t[p]=(t[p]+c*(r-l+1))%mod;return ;
}
pud(l,r,p);
int mid=l+r>>1;
if(x<=mid)up(l,mid,x,y,c,ls);
if(y>mid)up(mid+1,r,x,y,c,rs);
t[p]=(t[ls]+t[rs])%mod;
}
int query(int l,int r,int x,int y,int p){
if(x<=l&&r<=y){
return t[p];
}
pud(l,r,p);
int mid=l+r>>1;
int re=0;
if(x<=mid)re=(re+query(l,mid,x,y,ls))%mod;
if(y>mid)re=(re+query(mid+1,r,x,y,rs))%mod;
return re;
}
void insert(int x){
int y=1;
while(top[x]!=top[y]){
if(dep[top[x]]>dep[top[y]])swap(x,y);
up(1,n,idx[top[y]],idx[y],1,1);
y=fa[top[y]];
}if(dep[x]<dep[y])swap(x,y);
up(1,n,idx[y],idx[x],1,1);
}
LL ask(int x){
LL re=0;
int y=1;
while(top[x]!=top[y]){
if(dep[top[x]]>dep[top[y]])swap(x,y);
re+=query(1,n,idx[top[y]],idx[y],1);
re%=mod;
y=fa[top[y]];
}if(dep[x]<dep[y])swap(x,y);
re+=query(1,n,idx[y],idx[x],1);
re%=mod;
return re;
}
int main(){
scanf("%d%d",&n,&m);
int x;
for(int i=2;i<=n;i++){
scanf("%d",&x);x++;
add(x,i);add(i,x);
}
dfs1(1,0);
dfs2(1,1);
for(int i=1;i<=m;i++){
scanf("%d%d%d",&a[i].pos,&a[i+m].pos,&a[i].z);
a[i+m].pos++;a[i].z++;
a[i+m].z=a[i].z;a[i].id=a[i+m].id=i;a[i+m].flg=1;
}
sort(a+1,a+m+m+1,cmp);
int dir=0;
for(int i=1;i<=m+m;i++){
if(dir==a[i].pos){
ans[a[i].id+a[i].flg*m]=ask(a[i].z);continue;
}else{
for(int j=dir+1;j<=a[i].pos;j++){
insert(j);
}
ans[a[i].id+a[i].flg*m]=ask(a[i].z);dir=a[i].pos;
}
}
for(int i=1;i<=m;i++){
printf("%lld\n",(ans[i+m]-ans[i]+mod)%mod);
} }
/* 5 2
0
0
1
1
1 4 3
1 4 2
*/

BZOJ_3626_[LNOI2014]LCA_离线+树剖的更多相关文章

  1. BZOJ 3626 [LNOI2014]LCA:树剖 + 差分 + 离线【将深度转化成点权之和】

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3626 题意: 给出一个n个节点的有根树(编号为0到n-1,根节点为0,n <= 50 ...

  2. [LNOI2014]LCA(树剖+线段树)

    \(\%\%\% Fading\) 此题是他第一道黑题(我的第一道黑题是蒲公英) 一直不敢开,后来发现是差分一下,将询问离线,树剖+线段树维护即可 \(Code\ Below:\) #include ...

  3. bzoj 3626: [LNOI2014]LCA 离线+树链剖分

    3626: [LNOI2014]LCA Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 426  Solved: 124[Submit][Status] ...

  4. bzoj3626: [LNOI2014]LCA奇技淫巧+树剖+线段树

    题目求[a,b]到c的lca深度之和   显然是一个满足区间减法的操作 于是简化为 [1,b]到c的lca深度之和 (然并卵╮(╯▽╰)╭)然后就用奇技淫巧发现 a和b的lca深度=先把根节点到a的路 ...

  5. 【BZOJ3626】[LNOI2014]LCA 离线+树链剖分+线段树

    [BZOJ3626][LNOI2014]LCA Description 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深度定义为这个节点到根的距离+1.设dep[i]表示点i的深度 ...

  6. [BZOJ3626] [LNOI2014] LCA 离线 树链剖分

    题面 考虑到询问的\(l..r,z\)具有可减性,考虑把询问差分掉,拆成\(r,z\)和\(l-1,z\). 显然这些LCA一定在\(z\)到根的路径上.下面的问题就是怎么统计. 考虑不是那么暴力的暴 ...

  7. BZOJ 3626 [LNOI2014]LCA 树剖+(离线+线段树 // 在线+主席树)

    BZOJ 4012 [HNOI2015]开店 的弱化版,离线了,而且没有边权(长度). 两种做法 1 树剖+离线+线段树 这道题求的是一个点zzz与[l,r][l,r][l,r]内所有点的lcalca ...

  8. 【BZOJ 3626】 [LNOI2014]LCA【在线+主席树+树剖】

    题目链接: TP 题解:   可能是我比较纱布,看不懂题解,只好自己想了…… 先附一个离线版本题解[Ivan] 我们考虑对于询问区间是可以差分的,然而这并没有什么卵用,然后考虑怎么统计答案. 首先LC ...

  9. HDU 6162 Ch’s gift (树剖 + 离线线段树)

    Ch’s gift Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total S ...

随机推荐

  1. 使用padding后内容超出父级元素

    解决方法:

  2. Spring Boot + Jersey发生FileNotFoundException (No such file or directory)

    我在使用Spring Boot + Jersey 项目,解决了上一篇随笔中的FileNotFoundException,然后又报了一个FileNotFoundException,不过报错信息不一样了 ...

  3. SQL的几种连接:内连接、左联接、右连接、全连接、交叉连接

    SQL连接可以分为内连接.外连接.交叉连接. 数据库数据:             book表                                          stu表 1.内连接 ...

  4. linux swing 乱码

    转载 Linux下关于解决JavaSwing中文乱码的情况 redhed 貌似没出现乱码 本身就jdk就支持中文 红旗linux  suse等都不支持,需要自己手工配置,解决办法: 第一种方法: 1. ...

  5. wave音频格式一窥

    学习需要不断的刨根问底,首先,RIFF是微软多媒体格式的一个规范. 其次,有很多用C++实现wave解析的好文章.

  6. Activex、OLE、COM、OCX、DLL之间有什么区别?

    来源:http://www.blogjava.net/Jack2007/archive/2008/04/27/196392.html         熟悉面向对象编程和网络编程的人一定对ActiveX ...

  7. AI 学习之路

    前言:本文章纯属自己学习路线纪录,不喜勿喷. 最近AI很火,几乎是个程序员 都要去学习AI,作为一个菜鸡小前端,我也踏上了学习AI的方向. 在学习之中,最开始遇到了很多的困难,比如你不知道如何切入进来 ...

  8. 深度学习之 TensorFlow(二):TensorFlow 基础知识

    1.TensorFlow 系统架构: 分为设备层和网络层.数据操作层.图计算层.API 层.应用层.其中设备层和网络层.数据操作层.图计算层是 TensorFlow 的核心层. 2.TensorFlo ...

  9. 0基础一分钟入门Python

    这篇文章面向所有想学python的小伙伴(甚至你从没听过编程),这篇文章将会带你以最快的速度入门python.赶快上车,时间来不及了... 一,下载和安装python 1.下载: 1.1 python ...

  10. Dubbo配置引发的一个问题--- Duplicate spring bean id

    1.原因 因项目业务需要,要调用RPC框架,项目原本已经依赖了很多RPC接口需要启动时加载,所以准备做成启动时不预加载. 就是在配置的时候加上check=false. 官方文档解释的作用,就是Dubb ...