2049: [Sdoi2008]Cave 洞穴勘測 Time Limit: 10 Sec Memory Limit: 259 MB Description 輝輝熱衷于洞穴勘測。某天,他按照地圖來到了一片被標記為JSZX的洞穴群地區。經過初步勘測,輝輝發現這片區域由n個洞穴(分別編號為1到n)以及若干通道組成,并且每條通道連接了恰好兩個洞穴。假如兩個洞穴可以通過一條或者多條通道按一定順序連接起來,那么這兩個洞穴就是連通的,按順序連接在一起的這些通道則被稱之為這兩個洞穴之間的一條路徑。洞穴都十分堅固無法破壞,然而通道不太穩定,時常因為外界影響而發生改變,比如,根據有關儀器的監測結果,123號洞穴和127號洞穴之間有時會出現一條通道,有時這條通道又會因為某種稀奇古怪的原因被毀。輝輝有一臺監測儀器可以實時將通道的每一次改變狀況在輝輝手邊的終端機上顯示:如果監測到洞穴u和洞穴v之間出現了一條通道,終端機上會顯示一條指令 Connect u v 如果監測到洞穴u和洞穴v之間的通道被毀,終端機上會顯示一條指令 Destroy u v 經過長期的艱苦卓絕的手工推算,輝輝發現一個奇怪的現象:無論通道怎么改變,任意時刻任意兩個洞穴之間至多只有一條路徑。因而,輝輝堅信這是由于某種本質規律的支配導致的。因而,輝輝更加夜以繼日地堅守在終端機之前,試圖通過通道的改變情況來研究這條本質規律。然而,終于有一天,輝輝在堆積成山的演算紙中崩潰了……他把終端機往地面一砸(終端機也足夠堅固無法破壞),轉而求助于你,說道:“你老兄把這程序寫寫吧”。輝輝希望能隨時通過終端機發出指令 Query u v,向監測儀詢問此時洞穴u和洞穴v是否連通。現在你要為他編寫程序回答每一次詢問。已知在第一條指令顯示之前,JSZX洞穴群中沒有任何通道存在。 Input 第一行為兩個正整數n和m,分別表示洞穴的個數和終端機上出現過的指令的個數。以下m行,依次表示終端機上出現的各條指令。每行開頭是一個表示指令種類的字符串s(”Connect”、”Destroy”或者”Query”,區分大小寫),之后有兩個整數u和v (1≤u, v≤n且u≠v) 分別表示兩個洞穴的編號。 Output 對每個Query指令,輸出洞穴u和洞穴v是否互相連通:是輸出”Yes”,否則輸出”No”。(不含雙引號) Sample Input 樣例輸入1 cave.in 200 5 Query 123 127 Connect 123 127 Query 123 127 Destroy 127 123 Query 123 127 樣例輸入2 cave.in 3 5 Connect 1 2 Connect 3 1 Query 2 3 Destroy 1 3 Query 2 3 Sample Output 樣例輸出1 cave.out No Yes No 樣例輸出2 cave.out Yes No HINT 數據說明 10%的數據滿足n≤1000, m≤20000 20%的數據滿足n≤2000, m≤40000 30%的數據滿足n≤3000, m≤60000 40%的數據滿足n≤4000, m≤80000 50%的數據滿足n≤5000, m≤100000 60%的數據滿足n≤6000, m≤120000 70%的數據滿足n≤7000, m≤140000 80%的數據滿足n≤8000, m≤160000 90%的數據滿足n≤9000, m≤180000 100%的數據滿足n≤10000, m≤200000 保證所有Destroy指令將摧毀的是一條存在的通道本題輸入、輸出規模比較大,建議c/c++選手使用scanf和PRintf進行I/O操作以免超時
/*LCT裸題.然而我并不理解. */#include<iostream>#include<cstdio>#define MAXN 10001using namespace std;int n,m,fa[MAXN],tree[MAXN][2],st[MAXN],rev[MAXN];int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();} while(ch>='0'&&ch<='9') x=x*10+ch-48,ch=getchar(); return x*f;}bool isroot(int x){ return tree[fa[x]][0]!=x&&tree[fa[x]][1]!=x;}void pushdown(int x){ rev[tree[x][0]]^=1;rev[tree[x][1]]^=1;rev[x]^=1; swap(tree[x][0],tree[x][1]);return ;}void rotate(int x){ int y=fa[x],z=fa[y],l,r; if(tree[y][0]==x) l=0;else l=1;r=l^1; if(!isroot(y)){ if(tree[z][0]==y) tree[z][0]=x; else tree[z][1]=x; } fa[x]=z,fa[y]=x,fa[tree[x][r]]=y; tree[y][l]=tree[x][r];tree[x][r]=y; return ;}void splay(int x){ int top=0,y,z;st[++top]=x; for(int i=x;!isroot(i);i=fa[i]) st[++top]=fa[i]; for(int i=top;i;i--) if(rev[st[i]]) pushdown(st[i]); while(!isroot(x)) { y=fa[x],z=fa[y]; if(!isroot(y)) { if((tree[z][0]==y)^(tree[y][0]==x)) rotate(x); else rotate(y); } rotate(x); } return ;}void access(int x){ int t=0; while(x) splay(x),tree[x][1]=t,t=x,x=fa[x]; return ;}void makeroot(int x){ access(x),splay(x),rev[x]^=1; return ;}void join(int x,int y){ makeroot(x),fa[x]=y,splay(x); return ;}void cut(int x,int y){ makeroot(x),access(y),splay(y); tree[y][0]=fa[x]=0; return ;}int find(int x){ access(x),splay(x); while(tree[x][0]) x=tree[x][0]; return x;}int main(){ int x,y;char ch[10]; n=read(),m=read(); for(int i=1;i<=m;i++) { scanf("%s",ch);x=read(),y=read(); if(ch[0]=='C') join(x,y); else if(ch[0]=='D') cut(x,y); else find(x)==find(y)?printf("Yes/n"):printf("No/n"); } return 0;}新聞熱點
疑難解答