動態樹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和。
3 3 1 2 3 1 1 2 0 1 2 0 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 }