洛谷P4092树——并查集

时间:2020-12-06 09:12:58

题目:https://www.luogu.org/problemnew/show/P4092

利用并查集,倒序离线,那么从倒序来看被撤销标记的点就再也不会被标记,所以用并查集跳过;

莫名其妙的WA,调了一晚上,好像是dfs的地方有问题,莫名其妙的;注释掉的是WA的,现有的可以A,不知怎的。

代码如下:

#include<iostream>
#include<cstdio>
using namespace std;
int const MAXN=;
int n,q,head[MAXN],ct,fa[MAXN],ans[MAXN],as,cnt[MAXN],sta[MAXN],f[MAXN];
bool b[MAXN];
struct N{
int to,next;
N(int t=,int n=):to(t),next(n) {}
}edge[MAXN<<];
int find(int x)
{
if(x==fa[x])return x;
return fa[x]=find(fa[x]);
}
//void dfs(int x)
//{
// for(int i=head[x];i;i=edge[i].next)
// {
// int u=edge[i].to;
// if(u==f[x])continue;
//// if(!fa[u])fa[u]=x;
// f[u]=x;
// dfs(u);
// }
//}
void dfs(int u,int x)
{
f[x]=u;
for(int i=head[x];i;i=edge[i].next)
{
int v=edge[i].to;
if(v==u)continue;
dfs(x,v);
}
}
int main()
{
scanf("%d%d",&n,&q);
for(int i=;i<n;i++)
{
int x,y;
scanf("%d%d",&x,&y);
edge[++ct]=N(y,head[x]);head[x]=ct;
edge[++ct]=N(x,head[y]);head[y]=ct;
}
for(int i=;i<=q;i++)
{
char dc[];
cin>>dc;
cin>>sta[i];
if(dc[]=='C')cnt[sta[i]]++,b[i]=;
}
dfs(,);
cnt[]=;
for(int i=;i<=n;i++)//!
{
if(cnt[i])fa[i]=i;
else fa[i]=f[i];
}
for(int k=q;k;k--)
{
int x=sta[k];
if(b[k]==)ans[++as]=find(x);
else
{
cnt[x]--;
// if(!cnt[x])fa[x]=find(f[x]);
if(!cnt[x])fa[x]=f[x];//
}
}
for(int i=as;i;i--)
printf("%d\n",ans[i]);
return ;
}