LUOGU P1903 [国家集训队]数颜色 / 维护队列

传送门

 

解题思路

带修莫队,第一次写,其实和普通莫队差不多,就是多了个时间轴,块分n^(2/3)最优,时间复杂度O(n^(5/3))。

 

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>

using namespace std;
const int MAXN = 50005;

inline int rd(){
    int x=0,f=1;char ch=getchar();
    while(!isdigit(ch)) {f=ch=='-'?0:1;ch=getchar();}
    while(isdigit(ch))  {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
    return f?x:-x;
}

struct Data{
    int id,l,r,t;
}q[MAXN];

struct Upd{
    int pos,val;
}c[MAXN];

int n,m,bl[MAXN],siz,ans[MAXN],a[MAXN];
int Qcnt,Ccnt,cnt[1000005];

inline bool cmp(Data A,Data B){
    if(bl[A.l]!=bl[B.l]) return bl[A.l]<bl[B.l];
    if(bl[A.r]!=bl[B.r]) return bl[A.r]<bl[B.r];
    return A.t<B.t;
}

int main(){
    n=rd(),m=rd();int x,y;char s;siz=pow(n,(double)2/3);
    for(int i=1;i<=n;i++) a[i]=rd(),bl[i]=i/siz+1;
    for(int i=1;i<=m;i++){
        cin>>s;x=rd(),y=rd();
        if(s=='Q') 
            q[++Qcnt].l=x,q[Qcnt].r=y,q[Qcnt].id=Qcnt,q[Qcnt].t=Ccnt;
        else 
            c[++Ccnt].pos=x,c[Ccnt].val=y;
    }
    sort(q+1,q+1+Qcnt,cmp);
    int L=1,R=0,now=0,Ans=0;
    for(int i=1;i<=Qcnt;i++){
        while(L<q[i].l) {if(!(--cnt[a[L]])) Ans--;L++;}
        while(L>q[i].l) {L--;if(++cnt[a[L]]==1) Ans++;}
        while(R<q[i].r) {R++;if(++cnt[a[R]]==1) Ans++;}
        while(R>q[i].r) {if(!(--cnt[a[R]])) Ans--;R--;}
        while(now<q[i].t) {
//            cout<<q[i].t<<endl;
            now++;
            if(q[i].l<=c[now].pos && c[now].pos<=q[i].r){
                if(!(--cnt[a[c[now].pos]])) Ans--;
                if(++cnt[c[now].val]==1)  Ans++; 
            }
            swap(c[now].val,a[c[now].pos]);
        }
        while(now>q[i].t) {
            if(q[i].l<=c[now].pos && c[now].pos<=q[i].r) {
                if(!(--cnt[a[c[now].pos]])) Ans--;
                if(++cnt[c[now].val]==1)  Ans++; 
             }
             swap(c[now].val,a[c[now].pos]);
             now--;
        }
        ans[q[i].id]=Ans;
    }
    for(int i=1;i<=Qcnt;i++) printf("%d\n",ans[i]);
    return 0;
}
View Code

 

转载于:https://www.cnblogs.com/sdfzsyq/p/9687098.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值