LCT
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 | #include<cstdio> #include<algorithm> using namespace std; int n,m; struct Node{ int rev; Node *ch[2],*fa; Node(Node *fat); void Pushdown(); }*root[10005],*Null; Node::Node(Node *fat){ fa=fat; ch[0]=ch[1]=Null; rev=0; } void Node::Pushdown(){ if (rev){ if (ch[0]!=Null)ch[0]->rev^=1; if (ch[1]!=Null)ch[1]->rev^=1; swap(ch[0],ch[1]); rev=0; } } int Notroot(Node *x){ return (x->fa->ch[0]==x)||(x->fa->ch[1]==x); } void Prepare(Node *x){ if (Notroot(x))Prepare(x->fa); x->Pushdown(); } void Rotate(Node *x, int kind){ Node *y=x->fa,*z=y->fa; y->ch[!kind]=x->ch[kind]; if (x->ch[kind]!=Null)x->ch[kind]->fa=y; x->fa=z; if (Notroot(y))z->ch[z->ch[1]==y]=x; y->fa=x; x->ch[kind]=y; } void Splay(Node *x){ Prepare(x); while (Notroot(x)){ Node *y=x->fa,*z=y->fa; if (!Notroot(y))Rotate(x,y->ch[0]==x); else { if (y->ch[0]==x && z->ch[1]==y){Rotate(x,1);Rotate(x,0);} else if (y->ch[1]==x && z->ch[1]==y){Rotate(y,0);Rotate(x,0);} else if (y->ch[0]==x && z->ch[0]==y){Rotate(y,1);Rotate(x,1);} else {Rotate(x,0);Rotate(x,1);} } } } void Access(Node *x){ for (Node *y=Null;x!=Null;y=x,x=x->fa){Splay(x);x->ch[1]=y;} } void Makeroot(Node *x){ Access(x); Splay(x); x->rev^=1; } void Link(Node *u,Node *v){ Makeroot(u); u->fa=v; } void Cut(Node *u,Node *v){ Makeroot(u); Access(v); Splay(v); if (v->ch[0]==u){v->ch[0]=Null;u->fa=Null;} } Node *Find(Node *x){ Access(x); Splay(x); while (x->ch[0]!=Null)x=x->ch[0]; return x; } Node *GetNull(){ Node *p= new Node(Null); p->ch[0]=p->ch[1]=p->fa=p; p->rev=0; return p; } Node *ToLct( int x){ return root[x]; } int main(){ freopen ( "2049.in" , "r" ,stdin); freopen ( "2049.out" , "w" ,stdout); scanf ( "%d %d" ,&n,&m); Null=GetNull(); for ( int i=1;i<=n;i++)root[i]= new Node(Null); for ( int i=1;i<=m;i++){ int u,v; char s[10]; scanf ( "%s %d %d" ,s,&u,&v); if (s[0]== 'C' )Link(ToLct(u),ToLct(v)); if (s[0]== 'D' )Cut(ToLct(u),ToLct(v)); if (s[0]== 'Q' ){ if (Find(ToLct(u))==Find(ToLct(v))) puts ( "Yes" ); else puts ( "No" ); } } return 0; } |