可以删点的并查集

如题。

方法一:LCT!细节挺多,略。

方法二:如题(废话。。)

如果照传统的方法,比如1,2,3在一起要把1删掉,要保证1的爸爸和2,3以后不一样,如果1不是根节点就直接$fa[1]=1$,否则需要改所有的儿子。

上面的问题在于删掉根节点的操作过于冗杂,也就是说想一个办法使得1-n中没有一个点作为根节点,那就新申请n+1-2n做1-n的根,这样就不可能会有谁做根节点。

慢着,如果某个点被删掉,他的爸爸要变,又不能变成自己,咋整?再申请!

以下是:给一个图,保证每个点度数不超过10,一开始所有点都下线,每次可以选一个点上线和周围上线的点所在集合合并,或者选一个点下线,或者查某个点所在集合的大小。

 1 #include<string.h>
 2 #include
 3 #include
 4 #include
 5 #include
 6 //#include
 7 //#include
 8 using namespace std;
 9 
10 int n,m,q;
11 #define maxn 200011
12 struct Edge{int to,next;}edge[maxn<<1]; int first[maxn],le=2;
13 void in(int x,int y) {Edge &e=edge[le]; e.to=y; e.next=first[x]; first[x]=le++;}
14 void insert(int x,int y) {in(x,y); in(y,x);}
15 
16 int ufs[maxn*3],size[maxn*3]; bool live[maxn];
17 int find(int x) {return x==ufs[x]?x:(ufs[x]=find(ufs[x]));}
18 void Union(int x,int y)
19 {
20     x=find(x),y=find(y); if (x==y) return;
21     ufs[x]=y; size[y]+=size[x];
22 }
23 
24 int main()
25 {
26     scanf("%d%d%d",&n,&m,&q);
27     for (int i=1,x,y;i<=m;i++)
28     {
29         scanf("%d%d",&x,&y);
30         insert(x,y);
31     }
32     for (int i=1;i<=n;i++) ufs[i]=i+n,ufs[i+n]=i+n,size[i+n]=1; int tot=n+n;
33     char id;int x;
34     while (q--)
35     {
36         while ((id=getchar())!='i' && id!='q' && id!='o'); scanf("%d",&x);
37         if (id=='i')
38         {
39             live[x]=1;
40             for (int i=first[x];i;i=edge[i].next)
41             {
42                 const Edge &e=edge[i]; if (!live[e.to]) continue;
43                 Union(x,e.to);
44             }
45         }
46         else if (id=='o')
47         {
48             live[x]=0; int y=find(x); size[y]--;
49             ufs[x]=++tot; ufs[tot]=tot; size[tot]=1;
50         }
51         else
52         {
53             x=find(x);
54             printf("%d\n",size[x]);
55         }
56     }
57     return 0;
58 }
View Code

 

转载于:https://www.cnblogs.com/Blue233333/p/8340878.html

你可能感兴趣的:(可以删点的并查集)