這樣查詢另外一個點時也會有問題,那麼限定查詢區間爲dfs序兩端便可,而後這個題就很簡單了,而後這個問題就轉化成了$(size[root]-size[u])*size[u]$。get
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
inline 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-'0'; ch=getchar();}
return x*f;
}
#define MAXN 200010
int N,Q;
struct EdgeNode{
int next,to;
}edge[MAXN];
int head[MAXN],cnt=1;
inline void AddEdge(int u,int v) {cnt++; edge[cnt].next=head[u]; head[u]=cnt; edge[cnt].to=v;}
inline void InsertEdge(int u,int v) {AddEdge(u,v); AddEdge(v,u);}
int fa[MAXN];
inline int F(int x) {if (fa[x]==x) return x; else return fa[x]=F(fa[x]);}
struct SgtNode{
int lson,rson,size;
}tree[MAXN*20];
int root[MAXN],sz;
inline void Update(int x) {tree[x].size=tree[tree[x].lson].size+tree[tree[x].rson].size;}
inline int Merge(int x,int y)
{
if (!x || !y) return x|y;
tree[x].size+=tree[y].size;
tree[x].lson=Merge(tree[x].lson,tree[y].lson);
tree[x].rson=Merge(tree[x].rson,tree[y].rson);
return x;
}
inline void Insert(int &x,int l,int r,int pos)
{
x=++sz;
if (l==r) {
tree[x].size=1;
return;
}
int mid=(l+r)>>1;
if (pos<=mid) Insert(tree[x].lson,l,mid,pos);
else Insert(tree[x].rson,mid+1,r,pos);
Update(x);
}
inline int Query(int x,int l,int r,int L,int R)
{
if (!x) return 0;
if (L<=l && R>=r) return tree[x].size;
int mid=(l+r)>>1,re=0;
if (L<=mid) re+=Query(tree[x].lson,l,mid,L,R);
if (R>mid) re+=Query(tree[x].rson,mid+1,r,L,R);
return re;
}
int pl[MAXN],pre[MAXN],pr[MAXN],dfn,deep[MAXN];
inline void DFS(int now,int last)
{
pl[now]=++dfn; pre[dfn]=now;
for (int i=head[now]; i; i=edge[i].next)
if (edge[i].to!=last)
deep[edge[i].to]=deep[now]+1,
DFS(edge[i].to,now);
pr[now]=dfn;
}
struct QNode{
int opt,x,y;
}Qr[MAXN];
int main()
{
N=read(),Q=read();
for (int i=1; i<=Q; i++) {
char opt[5]; scanf("%s",opt+1);
int x=read(),y=read();
if (opt[1]=='A') InsertEdge(x,y);
Qr[i]=(QNode){(opt[1]=='A'? 0:1),x,y};
}
for (int i=1; i<=N; i++) if (!deep[i]) DFS(i,0);
for (int i=1; i<=N; i++) fa[i]=i,Insert(root[i],1,N,pl[i]);
for (int i=1; i<=Q; i++) {
if (Qr[i].opt==0) {
int x=F(Qr[i].x),y=F(Qr[i].y);
if (deep[x]>deep[y]) swap(x,y);
root[x]=Merge(root[x],root[y]);
fa[y]=x;
} else {
int x=Qr[i].x,y=Qr[i].y,z;
if (deep[x]<deep[y]) swap(x,y); z=F(x);
int siz=Query(root[z],1,N,pl[x],pr[x]);
printf("%lld\n",1LL*(tree[root[z]].size-siz)*siz);
}
}
return 0;
}