【[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;
}
posted @ 2019-01-02 12:15  asuldb  阅读(121)  评论(0编辑  收藏  举报