题目描述
萧薰儿是古国的公主,平时的一大爱好是采花。
今天天气晴朗,阳光明媚,公主清晨便去了皇宫中新建的花园采花。
花园足够大,容纳了n朵花,花有c种颜色(用整数1-c表示),且花是排成一排的,以便于公主采花。公主每次采花后会统计采到的花的颜色数,颜色数越多她会越高兴!同时,她有一癖好,她不允许最后自己采到的花中,某一颜色的花只有一朵。为此,公主每采一朵花,要么此前已采到此颜色的花,要么有相当正确的直觉告诉她,她必能再次采到此颜色的花。
由于时间关系,公主只能走过花园连续的一段进行采花,便让女仆福涵洁安排行程。福涵洁综合各种因素拟定了m个行程,然后一一向你询问公主能采到多少朵花(她知道你是编程高手,定能快速给出答案!),最后会选择令公主最高兴的行程(为了拿到更多奖金!)。
输入格式
第一行四个空格隔开的整数n、c以及m。接下来一行n个空格隔开的整数,每个数在[1, c]间,第i个数表示第i朵花的颜色。接下来m行每行两个空格隔开的整数l和r(l ≤ r),表示女仆安排的行程为公主经过第l到第r朵花进行采花。
输出格式
共m行,每行一个整数,第i个数表示公主在女仆的第i个行程中能采到的花的颜色数。
输入输出样例
输入 #1复制
5 3 5
1 2 2 3 1
1 5
1 2
2 2
2 3
3 5
输出 #1复制
2
0
0
1
0
和HH的项链十分相似,但是这道题是出现过两次才会有贡献。
于是我们维护第一次出现的位置,和第二次出现的位置即可。然后我们将第一次出现的位置,插入到树状数组当中,然后我们看第一次出现在树状数组当中的有多少即可。
AC代码:
#pragma GCC optimize(2)
#include<bits/stdc++.h>
//#define int long long
#define lowbit(x) (x&(-x))
using namespace std;
const int N=2e6+10;
int n,c,m,lst1[N],lst2[N],d[N],a[N],res[N];
struct node{int l,r,id;}t[N];
int cmp(node a,node b){return a.r<b.r;}
inline void add(int x,int v){for(;x<=n;x+=lowbit(x)) d[x]+=v;}
inline int ask(int x){int res=0; for(;x;x-=lowbit(x)) res+=d[x]; return res;}
signed main(){
cin>>n>>c>>m;
for(int i=1;i<=n;i++) scanf("%d",&a[i]);
for(int i=1;i<=m;i++) scanf("%d %d",&t[i].l,&t[i].r),t[i].id=i;
sort(t+1,t+m+1,cmp); int j=1;
for(int i=1;i<=m;i++){
for(;j<=t[i].r;j++){
if(!lst1[a[j]]) lst1[a[j]]=j;
else{
if(!lst2[a[j]]){
lst2[a[j]]=j,add(lst1[a[j]],1);
}else{
add(lst1[a[j]],-1); lst1[a[j]]=lst2[a[j]];
add(lst2[a[j]],1); lst2[a[j]]=j;
}
}
}
res[t[i].id]=ask(t[i].r)-ask(t[i].l-1);
}
for(int i=1;i<=m;i++) printf("%d\n",res[i]);
return 0;
}