【题解】梦幻布丁

\(Question\)

题目大意:给一个颜色序列,待修改,求区间颜色段数。

\(Solution\)

考虑线段树合并。

对于每一个颜色建一个线段树,维护每个颜色出现的位置。同时维护区间的颜色段数,以及最左端、最右端的颜色位置。

于是,对于合并颜色\(x,y\).,·将它们对应线段树合并即可。合并时,当一树为空时,直接返回;当递归到叶子时,左右均为该处\(pos\).颜色段为1,返回即可。

对于插入时,与合并处理边界一样。

维护信息时,左端信息优先考虑左子树,右端信息优先考虑右子树,总信息将左右颜色段相加,但若两端相交处颜色相同,则需要-1.

修改时一同维护\(ans.\)即可。

(蒟蒻第一次写非权值线段树的线段树合并)

\(\text{Code:}\)

#include<bits/stdc++.h>
using namespace std;
#define int long long
const int MAXN=1e6+10;
int n,m,ans;
int rt[MAXN],ch[MAXN<<5][2],rub[MAXN<<5],tot,cnt;
int a[MAXN],ls[MAXN<<5],rs[MAXN<<5],ms[MAXN<<5];
inline int build(){return (cnt?rub[cnt--]:++tot);}
inline void del(int x){
	ch[x][0]=ch[x][1]=0;
	ls[x]=rs[x]=ms[x]=0;
	rub[++cnt]=x;return;
}
inline void pushup(int x){
	ls[x]=ls[ch[x][0]]?ls[ch[x][0]]:ls[ch[x][1]];
	rs[x]=rs[ch[x][1]]?rs[ch[x][1]]:rs[ch[x][0]];
	ms[x]=ms[ch[x][0]]+ms[ch[x][1]]-(rs[ch[x][0]]+1==ls[ch[x][1]]);
}
void modify(int &p,int l,int r,int pos){
	if(!p)p=build();
	if(l==r){
		ls[p]=rs[p]=pos,ms[p]=1;
		return;
	}
	int mid=l+r>>1;
	if(pos<=mid)modify(ch[p][0],l,mid,pos);
	else modify(ch[p][1],mid+1,r,pos);
	pushup(p);
}
int merge(int x,int y,int l,int r){
	if(!x||!y){return x+y;}
	if(l==r){
		ls[x]=rs[x]=l;ms[x]=1;
		del(y);return x;
	}
	int mid=l+r>>1;
	ch[x][0]=merge(ch[x][0],ch[y][0],l,mid);
	ch[x][1]=merge(ch[x][1],ch[y][1],mid+1,r);
	del(y);pushup(x);return x;
}
inline void solve(){printf("%lld\n",ans);}
void change(int x,int y){
	ans-=ms[rt[x]];ans-=ms[rt[y]]; 
	rt[y]=merge(rt[y],rt[x],1,n);
	ans+=ms[rt[y]];rt[x]=0;
}
signed main(){
	scanf("%lld%lld",&n,&m);
	for(int i=1;i<=n;++i)scanf("%lld",&a[i]);
	for(int i=1;i<=n;++i){
		ans-=ms[rt[a[i]]];
		modify(rt[a[i]],1,n,i);
		ans+=ms[rt[a[i]]];
	}
	int Q=m;
	for(;Q;Q--){
		int opt;
		scanf("%lld",&opt);
		if(opt==2)solve();
		else {
			int x,y;
			scanf("%lld%lld",&x,&y);
			if(x==y)continue;
			change(x,y);
		}
	}
	return 0;
} 
posted @ 2020-03-28 14:00  Refined_heart  阅读(175)  评论(0编辑  收藏  举报