2821: 作詩(Poetize)
Time Limit: 50 Sec Memory Limit: 128 MBSubmit: 3356 Solved: 982
[Submit][Status][Discuss]
Description
Input
Output
輸出共m行,每行一個整數,第i個數表示SHY第i次能選出的漢字的最多種類數。
Sample Input
1 2 2 3 1
0 4
1 2
2 2
2 3
3 5
Sample Output
0
0
0
1
HINT
對於100%的數據,1<=n,c,m<=10^5
這道題的解法很類似區間衆數問題
先預處理
得到每兩個塊之間的答案
得到塊之間出現次數的前綴和 或在詢問過程中用二分求區間出現次數
每次詢問
ans=中間塊貢獻+兩側的貢獻
寫的O(nsqrt(n))爲啥跑的這麼慢啊 wwwww
#include<cmath>
#include<ctime>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<complex>
#include<iostream>
#include<algorithm>
#include<iomanip>
#include<vector>
#include<bitset>
#include<string>
#include<queue>
#include<map>
#include<set>
using namespace std;
typedef long long ll;
inline int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch<='9'&&ch>='0'){x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return x*f;
}
void print(int x)
{if(x<0)putchar('-'),x=-x;if(x>=10)print(x/10);putchar(x%10+'0');}
const int N=100100;
int n,a[N];
int block,block_num;
int bel[N],sum[N][320],g[320][320];
bool book[N];
void initial()
{
register int i,j,k,val;
block_num=1;
block=sqrt(n);
for(i=1,j=1;i<=n;++i,++j)
{
bel[i]=block_num;
if(j==block) j=0,block_num++;
}
for(i=1;i<=n;++i) sum[a[i]][bel[i]]++;
for(i=1;i<=n;++i)
if(!book[val=a[i]])
{
book[val]=1;
for(j=2;j<=block_num;++j)
sum[val][j]+=sum[val][j-1];
}
for(i=1;i<=n;++i) book[a[i]]=0;
for(i=1;i<=block_num;++i)
for(j=i;j<=block_num;++j)
{
g[i][j]=g[i][j-1];
for(k=(j-1)*block+1;k<=min(block*j,n);++k)
if(!book[val=a[k]])
{
book[val]=1;
g[i][j]+=( ((sum[val][j]-sum[val][i-1])&1)==0 );
if(j>i) g[i][j]-=( (sum[val][j-1]-sum[val][i-1]) && ((sum[val][j-1]-sum[val][i-1])&1)==0 );
}
for(k=(j-1)*block;k<=min(block*j,n);++k) book[a[k]]=0;
}
}
int num[N],tmp[N];
int query(int l,int r)
{
int res(0);
if(bel[l]==bel[r])
{
for(int i=l;i<=r;++i) num[a[i]]++;
for(int i=l,val;i<=r;++i)
if(!book[val=a[i]])
book[val]=1,res+=!(num[val]&1);
for(int i=l;i<=r;++i) book[a[i]]=0,num[a[i]]=0;
return res;
}
res=g[bel[l]+1][bel[r]-1];
int tot(0);
for(int i=l;i<=bel[l]*block;++i) tmp[++tot]=a[i];
for(int i=(bel[r]-1)*block+1;i<=r;++i) tmp[++tot]=a[i];
for(int i=1;i<=tot;++i) num[tmp[i]]++;
for(int i=1,pre,val;i<=tot;++i)
if(!book[val=tmp[i]])
{
book[val]=1;
pre=sum[val][bel[r]-1]-sum[val][bel[l]];
res+=!((pre+num[val])&1);
if(pre) res-=((pre&1)==0);
}
for(int i=1;i<=tot;++i) book[tmp[i]]=0,num[tmp[i]]=0;
return res;
}
int main()
{
n=read();
register int i,l,r,Q=read(),ans(0);
Q=read();
for(i=1;i<=n;++i) a[i]=read();
initial();
while(Q--)
{
l=(read()+ans)%n+1,r=(read()+ans)%n+1;
if(l>r) swap(l,r);
ans=query(l,r),print(ans),puts("");
}
return 0;
}