bzoj3435 [Wc2014]紫荊花之戀(動態點分治+替罪羊樹)

傳送門(權限)php

傳送門(非權限)ios

題解git

  我終終終終終終於作出來啦!!!數組

  做爲一個沒有學過替罪羊樹的蒟蒻現場學了一下替罪羊樹,做爲一個平衡樹都寫數組版本的看着大佬的指針題解無語只能硬去理解而後照着抄了一波指針ui

  而後怎麼作呢?spa

  先把題設式子變形一下$$dist(i,j)\leq r_i+r_j$$指針

  $$dist(i,LCA)+dist(LCA,j)\leq r_i+r_j$$code

  $$r_i-dist(i,LCA)\geq dist(j,LCA)-r_j$$blog

  而後咱們在每個點開兩棵平衡樹,分別維護以$i$爲根的子樹中$dist(i,u)-r_u$和$dist(fa[i],u)-r_u$的值。而後每一次跳點分樹時,記錄$r_i-dist(i,LCA)+1$,在平衡樹裏查詢有多少個數小於它就行了,修改直接往上跳,不斷改遞歸

  然而若是原樹是一條鏈怎麼辦?強制在線,必然會被卡成$O(n^2)$,怎麼辦?

  咱們聯想一下替罪羊樹的思想,若是點分樹上某一個點的子樹過大,直接拍扁重建。聯想一下替罪羊樹,能夠發現時間複雜度是能獲得保證的,這樣能夠保證時間複雜度是$O(nlogn)$。

  由於蒟蒻是第一次寫替罪羊樹&&第一次碼這麼長的代碼,因而加了一堆註釋,米娜應該可以看懂吧……

  1 // luogu-judger-enable-o2
  2 //minamoto
  3 #include<cstdio>
  4 #include<cstring>
  5 #include<iostream>
  6 #include<vector>
  7 #define ll long long
  8 #define inf 1000000000
  9 #define N 100005
 10 #define alpha 0.755
 11 using namespace std;
 12 #define getc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
 13 char buf[1<<21],*p1=buf,*p2=buf;
 14 template<class T>inline bool cmax(T&a,const T&b){return a<b?a=b,1:0;}
 15 inline int read(){
 16     #define num ch-'0'
 17     char ch;bool flag=0;int res;
 18     while(!isdigit(ch=getc()))
 19     (ch=='-')&&(flag=true);
 20     for(res=num;isdigit(ch=getc());res=res*10+num);
 21     (flag)&&(res=-res);
 22     #undef num
 23     return res;
 24 }
 25 char sr[1<<21],z[20];int C=-1,Z;
 26 inline void Ot(){fwrite(sr,1,C+1,stdout),C=-1;}
 27 inline void print(ll x){
 28     if(C>1<<20)Ot();if(x<0)sr[++C]=45,x=-x;
 29     while(z[++Z]=x%10+48,x/=10);
 30     while(sr[++C]=z[Z],--Z);sr[++C]='\n';
 31 }
 32 int n,e,head[N],Next[N<<1],ver[N<<1],val[N];
 33 inline void add(int u,int v){
 34     //加邊,構建原樹 
 35     ver[++e]=v,Next[e]=head[u],head[u]=e;
 36     ver[++e]=u,Next[e]=head[v],head[v]=e;
 37 }
 38 vector<int> to[N];
 39 int f[N][18],bin[25],tp,dep[N],len[N];
 40 inline int LCA(int a,int b){
 41     if(dep[a]<dep[b]) a^=b^=a^=b;
 42     int i,cha=dep[a]-dep[b];
 43     for(i=tp;~i;--i) if(cha&bin[i]) a=f[a][i];
 44     if(a==b) return a;
 45     for(i=tp;~i;--i) if(f[a][i]!=f[b][i]) a=f[a][i],b=f[b][i];
 46     return f[a][0];
 47 }
 48 inline int dis(int a,int b){return len[a]+len[b]-(len[LCA(a,b)]<<1);}
 49 struct Goat{
 50     int val,sz;Goat *ch[2];
 51     Goat(){}
 52     inline bool bad(){
 53         //判斷是否某個子樹過大 
 54         return ch[0]->sz>=sz*alpha+5||ch[1]->sz>=sz*alpha+5;
 55     }
 56 }*tree1[N],*tree2[N],mem[N<<8],*pool[N<<8],*null,*sta[N];
 57 int tot,top;
 58 void init(){
 59     //構建內存池,避免動態開點時間複雜度太大
 60     null=new Goat(); 
 61     null->ch[0]=null->ch[1]=null,null->val=null->sz=0;
 62     for(int i=0;i<(N<<8);++i) pool[i]=mem+i;
 63     tot=(N<<8)-1;
 64     for(int i=0;i<=n;++i) tree1[i]=tree2[i]=null;
 65 }
 66 Goat** insert(Goat *&a,int val){
 67     //插入節點,並判斷是否有子樹過大
 68     //注意要開引用 
 69     if(a==null){
 70         a=pool[tot--],a->ch[0]=a->ch[1]=null;
 71         a->val=val,a->sz=1;return &null;
 72     }
 73     ++a->sz;
 74     //小於等於往左插,大於往右插 
 75     Goat **o=insert(a->ch[a->val<val],val);
 76     if(a->bad()) o=&a;return o;
 77 }
 78 int getrk(Goat *o,int val){
 79     //查找有多少比val小的數 
 80     if(o==null) return 0;
 81     return (o->val>=val)?getrk(o->ch[0],val):(getrk(o->ch[1],val)+o->ch[0]->sz+1);
 82 }
 83 void Erholung(Goat *o){
 84     //清除節點,回收內存池 
 85     if(o==null) return;
 86     if(o->ch[0]!=null) Erholung(o->ch[0]);
 87     pool[++tot]=o;
 88     if(o->ch[1]!=null) Erholung(o->ch[1]);
 89 }
 90 void travel(Goat *o){
 91     //暴力重構整棵樹(遞歸找節點) 
 92     if(o==null) return;
 93     if(o->ch[0]!=null) travel(o->ch[0]);
 94     sta[++top]=o;
 95     if(o->ch[1]!=null) travel(o->ch[1]);
 96 }
 97 Goat* build(int l,int r){
 98     //重構 
 99     if(l>r) return null;
100     int mid=l+r>>1;
101     Goat *o=sta[mid];o->sz=r-l+1;
102     o->ch[0]=build(l,mid-1),o->ch[1]=build(mid+1,r);
103     return o;
104 }
105 inline void rebuild(Goat *&o){top=0,travel(o),o=build(1,top);};
106 inline void Insert(Goat *&a,int val){
107     //同,這裏和上面rebuild也要開引用
108     Goat **o=insert(a,val);
109     if(*o!=null) rebuild(*o);
110 }
111 int sz[N],son[N],size,rt,fa[N];bool vis[N];
112 void findrt(int u,int fa){
113     sz[u]=1,son[u]=0;
114     for(int i=head[u];i;i=Next[i]){
115         int v=ver[i];
116         if(v!=fa&&!vis[v]){
117             findrt(v,u),sz[u]+=sz[v],cmax(son[u],sz[v]);
118         }
119     }
120     cmax(son[u],size-sz[u]);
121     if(son[u]<son[rt]) rt=u;
122 }
123 void dfs(int u,int f,int rt){
124     //遍歷子樹,把全部的東西都插到平衡樹裏 
125     Insert(tree1[rt],dis(u,rt)-val[u]);
126     if(fa[rt]) Insert(tree2[rt],dis(u,fa[rt])-val[u]);
127     for(int i=head[u];i;i=Next[i]){
128         int v=ver[i];
129         if(v!=f&&!vis[v]) dfs(v,u,rt);
130     }
131 }
132 void solve(int u,int f){
133     fa[u]=f,vis[u]=1;
134     int totsz=size;
135     dfs(u,0,u);
136     for(int i=head[u];i;i=Next[i]){
137         int v=ver[i];
138         if(!vis[v]){
139             rt=0,size=sz[v]>sz[u]?totsz-sz[u]:sz[v];
140             findrt(v,0),to[u].push_back(rt),solve(rt,u);
141         }
142     }
143 }
144 void recover(int x){
145     //遍歷點分樹,清空節點 
146     ++size,vis[x]=0;
147     Erholung(tree1[x]),Erholung(tree2[x]);
148     tree1[x]=tree2[x]=null;
149     for(int i=0,k=to[x].size();i<k;++i) recover(to[x][i]);
150     to[x].clear();
151 }
152 void rebuild(int x){
153     //點分樹某一子樹過大,重構 
154     size=0,recover(x),rt=0,findrt(x,0);
155     if(fa[x])
156     for(int i=0,j=to[fa[x]].size();i<j;++i)
157     if(to[fa[x]][i]==x) to[fa[x]][i]=rt;
158     solve(rt,fa[x]);
159 }
160 ll ans=0;
161 int insert(int x){
162     register int i,ds,res=0;
163     //求出小於等於val[x]-dis(x,fa[i])的個數,只要在平衡樹裏找小於val[x]-dis(x,fa[i])+1的就能夠了 
164     for(i=x;fa[i];i=fa[i])
165     ds=val[x]-dis(x,fa[i])+1,ans+=getrk(tree1[fa[i]],ds)-getrk(tree2[i],ds);
166     Insert(tree1[x],-val[x]);
167     //而後維護修改 
168     for(i=x;fa[i];i=fa[i]){
169         int dist=dis(fa[i],x)-val[x];
170         Insert(tree1[fa[i]],dist);
171         Insert(tree2[i],dist);
172     }
173     //考慮是否須要拍扁重建 
174     for(i=x;fa[i];i=fa[i])
175     if(tree1[i]->sz>=tree1[fa[i]]->sz*alpha+5) res=fa[i];
176     return res;
177 }
178 int main(){
179     n=read(),n=read();
180     register int i,j,b,x;
181     for(bin[0]=i=1;i<=20;++i) bin[i]=bin[i-1]<<1;
182     while(bin[tp+1]<=n) ++tp;
183     son[0]=n+1,rt=0,init();
184     for(int i=1;i<=n;++i){
185         fa[i]=f[i][0]=read()^(ans%inf),b=read(),val[i]=read();
186         dep[i]=dep[f[i][0]]+1,len[i]=len[f[i][0]]+b,vis[i]=1;
187         if(fa[i]) to[fa[i]].push_back(i),add(f[i][0],i);
188         for(j=1;bin[j]+1<=dep[i];++j) f[i][j]=f[f[i][j-1]][j-1];
189         x=insert(i);if(x) rebuild(x);
190         print(ans);
191     }
192     Ot();
193     return 0;
194 }
相關文章
相關標籤/搜索