【做题记录】P4211 [LNOI2014]LCA

P4211 [LNOI2014]LCA

题意

给出一个 \(n\) 个节点的有根树(编号为 \(0\)\(n-1\),根节点为 \(0\))。

一个点的深度定义为这个节点到根的距离 \(+1\)

\(dep[i]\) 表示点 \(i\) 的深度,\(LCA(i,j)\) 表示 \(i\)\(j\) 的最近公共祖先。

\(m\) 次询问,每次询问给出 l r z,求 \(\sum_{i=l}^r dep[LCA(i,z)]\)

题解

我们考虑这样一件事情,就是说,两个点的 \(Lca\) 的深度的意义到底是什么,如何才能够快速求的?

在我们不会求 \(Lca\) 的时候,我们会将一个点一个一个往上爬,并将路径上的点染色。之后我们将另一个点暴力往上跳,第一个染色过的点就是 \(Lca\)

我们考虑到这样一件事情,如果我们将一个点到根的路径上每个点数值加 \(1\),再求得另一个点到根的路径上的权值和,这不就是 \(Lca\) 的深度了吗?

于是求解一次询问就相当于对于 \(l\le x\le r\),将 \(x\) 到根的路径上每个点加一,最后答案就是 \(z\) 到根的权值和。

由于询问次数很多,不可能每次都这样处理一遍,考虑将询问拆开。

\[\sum_{i=l}^r dep[LCA(i,z)]=\sum_{i=1}^r dep[LCA(i,z)]-\sum_{i=1}^{l-1} dep[LCA(i,z)] \]

这样我们就可以在 \(O(n\log n)\) 复杂度内解决问题啦!

这就叫数据结构二步曲,学习一下,代码我现在放下。

#define Maxn 50005
#define pb push_back
#define mod 201314
typedef long long ll;
int n,m,tot,Time,cnt;
int ans[Maxn];
int dfn[Maxn],tp[Maxn],fa[Maxn],dep[Maxn],siz[Maxn],bigson[Maxn];
int hea[Maxn],nex[Maxn],ver[Maxn];
struct TREE { int laz,sum; }tree[Maxn<<2];
struct Query { int num,z,opt; };
vector<Query> q[Maxn];
void dfs1(int x)
{
	 siz[x]=1;
	 for(int i=hea[x];i;i=nex[i])
	 {
	 	 fa[ver[i]]=x,dep[ver[i]]=dep[x]+1;
	 	 dfs1(ver[i]),siz[x]+=siz[ver[i]];
	 	 if(siz[ver[i]]>siz[bigson[x]]) bigson[x]=ver[i];
	 }
}
void dfs2(int x,int T)
{
	 tp[x]=T,dfn[x]=++Time;
	 if(bigson[x]) dfs2(bigson[x],T);
	 for(int i=hea[x];i;i=nex[i])
	 {
	 	 if(ver[i]==bigson[x]) continue;
	 	 dfs2(ver[i],ver[i]);
	 }
}
void pushdown(int p,int nl,int nr)
{
	 int mid=(nl+nr)>>1;
	 tree[p<<1].laz=(tree[p<<1].laz+tree[p].laz)%mod;
	 tree[p<<1|1].laz=(tree[p<<1|1].laz+tree[p].laz)%mod;
	 tree[p<<1].sum=(tree[p<<1].sum+1ll*tree[p].laz*(mid-nl+1)%mod)%mod;
	 tree[p<<1|1].sum=(tree[p<<1|1].sum+1ll*tree[p].laz*(nr-mid)%mod)%mod;
	 tree[p].laz=0;
}
void add(int p,int nl,int nr,int l,int r)
{
	 if(nl>=l && nr<=r) { tree[p].laz++,tree[p].sum+=(nr-nl+1); return; }
	 pushdown(p,nl,nr);
	 int mid=(nl+nr)>>1;
	 if(mid>=l) add(p<<1,nl,mid,l,r);
	 if(mid<r) add(p<<1|1,mid+1,nr,l,r);
	 tree[p].sum=(tree[p<<1].sum+tree[p<<1|1].sum)%mod;
}
int query(int p,int nl,int nr,int l,int r)
{
	 if(nl>=l && nr<=r) return tree[p].sum;
	 pushdown(p,nl,nr);
	 int mid=(nl+nr)>>1,ret=0;
	 if(mid>=l) ret=(ret+query(p<<1,nl,mid,l,r))%mod;
	 if(mid<r) ret=(ret+query(p<<1|1,mid+1,nr,l,r))%mod;
	 tree[p].sum=(tree[p<<1].sum+tree[p<<1|1].sum)%mod;
	 return ret;
}
inline void add_path(int x)
{
	 while(tp[x]!=tp[1]) add(1,1,n,dfn[tp[x]],dfn[x]),x=fa[tp[x]];
	 add(1,1,n,dfn[1],dfn[x]);
}
inline int query_path(int x)
{
	 int ret=0;
	 while(tp[x]!=tp[1]) ret=(ret+query(1,1,n,dfn[tp[x]],dfn[x]))%mod,x=fa[tp[x]];
	 ret=(ret+query(1,1,n,dfn[1],dfn[x]))%mod;
	 return ret;
}
inline void add_edge(int x,int y){ ver[++tot]=y,nex[tot]=hea[x],hea[x]=tot; }
int main()
{
	 n=rd(),m=rd();
	 for(int i=2,f;i<=n;i++) f=rd()+1,add_edge(f,i);
	 dfs1(1),dfs2(1,0); // 记得输入的下标从 0 开始 
	 for(int i=1,l,r,z;i<=m;i++)
	 {
	 	 l=rd()+1,r=rd()+1,z=rd()+1;
	 	 q[l-1].pb((Query){i,z,-1});
	 	 q[r].pb((Query){i,z,1});
	 }
	 for(int i=1;i<=n;i++)
	 {
	 	 add_path(i);
	 	 for(Query j:q[i])
		  	 ans[j.num]=(ans[j.num]+j.opt*query_path(j.z)%mod+mod)%mod;
	 }
	 for(int i=1;i<=m;i++) printf("%d\n",ans[i]);
	 return 0;
}
posted @ 2021-11-12 15:27  EricQian06  阅读(50)  评论(1编辑  收藏  举报