zoukankan      html  css  js  c++  java
  • CF1491H Yuezheng Ling and Dynamic Tree

    XXXII.CF1491H Yuezheng Ling and Dynamic Tree

    首先,相信大家都做过[HNOI2010]弹飞绵羊这道经典原题,而这题显然是那题的增强版。

    众所周知,该原题有两种常见解法:LCT或分块。凭直觉发现LCT不可能处理这种区间修改问题,于是我们考虑分块。

    分块的话,就需要维护一个位置在跳完它所在的那个块后的落点。但这里为了方便求LCA,所以我们借鉴树剖的思想,令 \(go_x\) 为离开块前最后一个到达的节点,令 \(len_x\) 表示其在块内跳的次数。

    我们发现,若一个块被修改了超过块长次,则其中所有元素的 \(go_x\) 都必定为其自身,\(len_x\) 都必定为 \(0\),就可以直接打 tag 维护里面所有元素的 \(a_x\) 了。

    于是对于每次修改,我们暴力重构未满的块,若块长是 \(\sqrt{n}\),则一共 \(\sqrt n\) 个块,每块被修改 \(\sqrt n\) 次,每次修改 \(\sqrt n\) 个元素,总复杂度 \(O(n\sqrt n)\),可以接受。

    下面考虑求LCA。明显,我们可以先跑出从 \(x,y\) 到根的路径上所有需要的点(指所有 \(go_x\) 及其父亲)的 \(dep\),然后就可以像树剖一样处理,直到两个东西的 \(go\) 相同。此时它们必在同一块内,直接暴力即可。此部分复杂度 \(O(\sqrt n)\)

    故总复杂度 \(O(n\sqrt n)\)

    代码:

    #include<bits/stdc++.h>
    using namespace std;
    const int BBB=500;
    int n,m,a[100100],BLK[100100],go[100100],len[100100],dep[100100],lp[510],tag[510],num;
    bool fin[510];
    void rebuild(int id,int L,int R,int x){
    	if(L<=lp[id]&&R>=lp[id+1]-1&&fin[id]){tag[id]+=x,tag[id]=min(tag[id],n);return;}
    	fin[id]=true;
    	for(int i=lp[id];i<lp[id+1];i++){
    		a[i]=max(0,a[i]-tag[id]);
    		if(L<=i&&i<=R)a[i]=max(0,a[i]-x);
    		if(a[i]<lp[id])go[i]=i,len[i]=0;else go[i]=go[a[i]],len[i]=len[a[i]]+1,fin[id]=false;
    	}
    	tag[id]=0;
    }
    #define fa(x) max(a[x]-tag[BLK[x]],0)
    int chain(int x){
    	if(x==0)return dep[x]=0;
    	dep[go[x]]=chain(fa(go[x]))+1;
    	if(!go[x])dep[go[x]]=0;
    	return dep[x]=dep[go[x]]+len[x];
    }
    int read(){
    	int x=0;
    	char c=getchar();
    	while(c>'9'||c<'0')c=getchar();
    	while(c>='0'&&c<='9')x=(x<<3)+(x<<1)+(c^48),c=getchar();
    	return x;
    }
    void modify(){
    	int L=read()-1,R=read()-1,x=read();
    	for(int i=BLK[L];i<=BLK[R];i++)rebuild(i,L,R,x);
    }
    bool lab[100100];
    void LCA(){
    //	printf("BEF:%d\n",dep[0]);
    	int x=read()-1,y=read()-1;
    	chain(x),chain(y);
    //	printf("AFT:%d\n",dep[0]);
    	while(go[x]!=go[y]){
    		if(dep[go[x]]<dep[go[y]])swap(x,y);
    		x=fa(go[x]);
    	}
    	lab[0]=true;
    	for(int i=x;i&&BLK[i]==BLK[x];i=fa(i))lab[i]=true;
    	for(int i=y;BLK[i]==BLK[y];i=fa(i))if(lab[i]){printf("%d\n",i+1);break;}
    	for(int i=x;i&&BLK[i]==BLK[x];i=fa(i))lab[i]=false;
    }
    int main(){
    //	freopen("I.in","r",stdin);
    	n=read(),m=read(),a[0]=-1;
    	for(int i=1;i<n;i++)a[i]=read()-1,BLK[i]=i/BBB;
    	lp[num=BLK[n]=BLK[n-1]+1]=n;for(int i=0;i<num;i++)lp[i]=i*BBB;
    	for(int i=0;i<num;i++)rebuild(i,-1,-1,0);
    	for(int i=1;i<=m;i++)if(read()==1)modify();else LCA();
    	return 0;
    }
    

  • 相关阅读:
    Arr
    class4
    class3联大网页
    class33
    class3
    人机交换 NO 1书签
    大数据的框架与特点
    mapreduce排序
    mapreduce求平均数
    mapreduce去重
  • 原文地址:https://www.cnblogs.com/Troverld/p/14611600.html
Copyright © 2011-2022 走看看