列表

详情


NC210197. 维护队列

描述

你小时候玩过弹珠吗?
小朋友A有一些弹珠,A喜欢把它们排成队列,从左到右编号为1到N。为了整个队列鲜艳美观,小朋友想知道某一段连续弹珠中,不同颜色的弹珠有多少。当然,A有时候会依据个人喜好,替换队列中某个弹珠的颜色。但是A还没有学过编程,且觉得头脑风暴太浪费脑力了,所以向你来寻求帮助。

输入描述

输入文件第一行包含两个整数N和M。
第二行N个整数,表示初始队列中弹珠的颜色。
接下来M行,每行的形式为“Q L R”或“R x c”,“Q L R”表示A想知道从队列第L个弹珠到第R个弹珠中,一共有多少不同颜色的弹珠,“R x c”表示A把x位置上的弹珠换成了c颜色。

输出描述

对于每个Q操作,输出一行表示询问结果。

示例1

输入:

2 3
1 2
Q 1 2
R 1 2
Q 1 2

输出:

2
1

原站题解

上次编辑到这里,代码来自缓存 点击恢复默认模板

C++(clang++11) 解法, 执行用时: 17ms, 内存消耗: 4632K, 提交时间: 2020-12-10 12:49:10

#include<bits/stdc++.h>
using namespace std;
#define LL long long
#define debug(x) cerr<<#x<<" = "<<x
#define sp <<"  "
#define el <<endl
#define fgx cerr<<" ----------------------------------- "<<endl
#define pii pair<int,int>
#define ULL unsigned long long
#define DB double
#define uint unsigned int
#define pb push_back
#define mpt make_pair
#define fr first
#define sc second
inline int read(){
	int nm=0,fh=1; char cw=getchar();
	for(;!isdigit(cw);cw=getchar()) if(cw=='-') fh=-1;
	for(;isdigit(cw);cw=getchar()) nm=nm*10+(cw-'0');
	return nm*fh;
}
#define M 200020
#define MAXN 1000020
int n,m,K,a[M],blg[M],res; pii sta[M];
int top,tail,cnt[MAXN],L,R,nowtms;
LL ans[M];
struct Query{int l,r,tms,id;}que[M];
inline bool cmp(Query a,Query b){
	if(blg[a.l]^blg[b.l]) return blg[a.l]<blg[b.l];
	if(blg[a.r]^blg[b.r]) return blg[a.r]<blg[b.r];
	return a.tms<b.tms;
}
inline void ins(int x){cnt[x]++; if(cnt[x]==1) res++;}
inline void ers(int x){--cnt[x]; if(!cnt[x]) --res;}
inline void chg(int now){
	int x=sta[now].fr;
	if(L<=x&&x<=R) ers(a[x]);
	swap(a[x],sta[now].sc);
	if(L<=x&&x<=R) ins(a[x]);
}

int main(){
	n=read(),m=read(),K=max((int)pow(n,0.666666),1);
	for(int i=1;i<=n;i++) a[i]=read(),blg[i]=(i-1)/K+1;
	for(int i=1,x,y;i<=m;i++){
		char op[15]; scanf("%s",op);
		if(op[0]=='Q') x=read(),y=read(),top++,que[top]=(Query){x,y,tail,top};
		else x=read(),y=read(),sta[++tail]=mpt(x,y);
	}
	sort(que+1,que+top+1,cmp);
	L=1,R=0,nowtms=0;
	for(int i=1;i<=top;i++){
		while(L>que[i].l) ins(a[--L]);
		while(R<que[i].r) ins(a[++R]);
		while(L<que[i].l) ers(a[L++]);
		while(R>que[i].r) ers(a[R--]);
		while(nowtms<que[i].tms) chg(++nowtms);
		while(nowtms>que[i].tms) chg(nowtms--);
		ans[que[i].id]=res;
	}
	for(int i=1;i<=top;i++) printf("%d\n",ans[i]);
	return 0;
}

上一题