[luogu 3690]Link Cut Tree

題目背景

動態樹ios

題目描述

給定N個點以及每一個點的權值,要你處理接下來的M個操做。操做有4種。操做從0到3編號。點從1到N編號。spa

0:後接兩個整數(x,y),表明詢問從x到y的路徑上的點的權值的xor和。保證x到y是聯通的。code

1:後接兩個整數(x,y),表明鏈接x到y,若x到Y已經聯通則無需鏈接。blog

2:後接兩個整數(x,y),表明刪除邊(x,y),不保證邊(x,y)存在。get

3:後接兩個整數(x,y),表明將點X上的權值變成Y。string

輸入輸出格式

輸入格式:it

第1行兩個整數,分別爲N和M,表明點數和操做數。io

第2行到第N+1行,每行一個整數,整數在[1,10^9]內,表明每一個點的權值。class

第N+2行到第N+M+1行,每行三個整數,分別表明操做類型和操做所需的量。stream

輸出格式:

對於每個0號操做,你須輸出X到Y的路徑上點權的Xor和。

輸入輸出樣例

輸入樣例#1: 複製
3 3 
1
2
3
1 1 2
0 1 2 
0 1 1
輸出樣例#1: 複製
3
1

說明

數據範圍: $1 \leq N, M \leq 3 \cdot {10}^5$

題解:

動態樹練手。

  1 //Never forget why you start
  2 #include<iostream>
  3 #include<cstdio>
  4 #include<cstdlib>
  5 #include<cstring>
  6 #include<cmath>
  7 #include<algorithm>
  8 #define ll(x) lct[x].child[0]
  9 #define rr(x) lct[x].child[1]
 10 #define son(x,t) lct[x].child[t]
 11 using namespace std;
 12 int n,m;
 13 int read(){
 14   int ans=0,f=1;char i=getchar();
 15   while(i<'0'||i>'9'){if(i=='-')f=-1;i=getchar();}
 16   while(i>='0'&&i<='9'){ans=ans*10+i-'0';i=getchar();}
 17   return ans*f;
 18 }
 19 struct LCT{
 20   int child[2],fa,x,size,sum,rev;
 21   bool is_root;
 22 }lct[300005];
 23 void push_up(int root){
 24   if(!root)return;
 25   lct[root].sum=lct[root].x^lct[ll(root)].sum^lct[rr(root)].sum;
 26   lct[root].size=lct[ll(root)].size+lct[rr(root)].size+1;
 27 }
 28 int getson(int x){
 29   return x==son(lct[x].fa,1);
 30 }
 31 void push_rev(int x){
 32   if(!x)return;
 33   swap(ll(x),rr(x));
 34   lct[x].rev^=1;
 35 }
 36 void push_down(int x){
 37   if(lct[x].rev&&x){
 38     push_rev(ll(x));
 39     push_rev(rr(x));
 40     lct[x].rev^=1;
 41   }
 42 }
 43 void push(int x){
 44   if(!lct[x].is_root)push(lct[x].fa);
 45   push_down(x);
 46 }
 47 void rotate(int x){
 48   if(lct[x].is_root)return;
 49   int fa=lct[x].fa,fafa=lct[fa].fa,t=getson(x);
 50   son(fa,t)=son(x,!t);if(son(x,!t))lct[son(x,!t)].fa=fa;
 51   lct[fa].fa=x;son(x,!t)=fa;
 52   lct[x].fa=fafa;
 53   if(!lct[fa].is_root)son(fafa,son(fafa,1)==fa)=x;
 54   else lct[x].is_root=1,lct[fa].is_root=0;
 55   push_up(fa);
 56   push_up(x);
 57 }
 58 void splay(int x){
 59   push(x);
 60   for(int fa;!lct[x].is_root;rotate(x))
 61     if(!lct[fa=lct[x].fa].is_root)
 62       rotate(getson(fa)==getson(x)?fa:x);
 63 }
 64 void access(int x){
 65   int y=0;
 66   while(x){
 67     splay(x);
 68     lct[rr(x)].is_root=1;
 69     lct[rr(x)=y].is_root=0;
 70     push_up(x);
 71     x=lct[y=x].fa;
 72   }
 73 }
 74 void mroot(int x){
 75   access(x);
 76   splay(x);
 77   push_rev(x);
 78 }
 79 void link(int u,int v){
 80   mroot(u);
 81   lct[u].fa=v;
 82 }
 83 void cut(int u,int v){
 84   mroot(u);
 85   access(v);splay(v);
 86   lct[son(v,0)].fa=lct[v].fa;
 87   lct[son(v,0)].is_root=1;
 88   ll(v)=lct[v].fa=0;
 89   push_up(v);
 90 }
 91 int find(int x){
 92   access(x);
 93   splay(x);
 94   if(ll(x))x=ll(x);
 95   return x;
 96 }
 97 int main(){
 98   int i,j;
 99   n=read();m=read();
100   for(i=1;i<=n;i++){
101     lct[i].x=lct[i].sum=read();
102     lct[i].size=1;
103     ll(i)=rr(i)=lct[i].fa=0;
104     lct[i].is_root=1;
105   }
106   int a,b,c;
107   for(i=1;i<=m;i++){
108     c=read();a=read();b=read();
109     if(c==0){
110       mroot(a);
111       access(b);
112       splay(b);
113       printf("%d\n",lct[b].sum);
114     }
115     else if(c==1){
116       int p=find(a),q=find(b);
117       if(p!=q)link(a,b);
118     }
119     else if(c==2){
120       mroot(a);
121       access(b);
122       splay(b);
123       if(ll(b)==a)cut(a,b);
124     }
125     else{
126       mroot(a);
127       splay(a);
128       lct[a].x=b;
129       push_up(a);
130     }
131   }
132   return 0;
133 }
相關文章
相關標籤/搜索