【[HEOI2012]采花】

\(HH\)的项链加强版,数据范围和题意都加强了

题意大概:给出n个数,求区间出现次数>=2的数的个数。

一眼莫队,可是我还不会莫队啊

那就树状数组吧

回忆一下\(HH\)的项链,套路差不多,那道题我们维护的是每一种颜色最后出现的位置,因为根据其最后出现的位置我们就可以判断其是否在区间里

而判断这道题也很简单,我们维护每一个颜色倒数第二次出现是在什么位置,这样就能判断其是否出现次数大于等于二了

还是离线+树状数组就可以啦

代码

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#define re register
#define lowbit(x) ((x)&(-x))
#define maxn 2000005
inline int read()
{
    char c=getchar();
    int x=0;
    while(c<'0'||c>'9') c=getchar();
    while(c>='0'&&c<='9')
        x=(x<<3)+(x<<1)+c-48,c=getchar();
    return x;
}
int c[2][maxn];
int n,m,T;
struct Ask
{
    int l,r,rk;
}q[maxn];
inline int cmp(Ask K,Ask M)
{
    return K.r<M.r;
}
inline void add(int x,int o,int val)
{
    for(re int i=x;i<=n;i+=lowbit(i))
        c[o][i]+=val;
}
inline int ask(int x,int o)
{
    int ans=0;
    for(re int i=x;i;i-=lowbit(i))
        ans+=c[o][i];
    return ans;
}
int a[maxn],pre[maxn],col_p[maxn],Ans[maxn];
int main()
{
    n=read(),T=read(),m=read();
    for(re int i=1;i<=n;i++) a[i]=read();
    for(re int i=1;i<=m;i++)
        q[i].l=read(),q[i].r=read(),q[i].rk=i;
    std::sort(q+1,q+m+1,cmp);
    for(re int i=1;i<=n;i++)
        pre[i]=col_p[a[i]],col_p[a[i]]=i;
    int now=1;
    for(re int i=1;i<=n;i++)
    {
        if(!pre[i]) add(i,0,1);
        else
        {
            if(!pre[pre[i]]) add(i,0,1),add(pre[i],1,1),add(pre[i],0,-1);
            else
            {
                add(i,0,1),add(pre[i],0,-1);
                add(pre[i],1,1),add(pre[pre[i]],1,-1);
            }
        }
        while(i==q[now].r) Ans[q[now].rk]=ask(q[now].r,1)-ask(q[now].l-1,1),now++;
    }
    for(re int i=1;i<=m;i++)
        printf("%d\n",Ans[i]);
    return 0;
}

猜你喜欢

转载自www.cnblogs.com/asuldb/p/10207886.html
今日推荐