六一儿童节到了, SHUXK 被迫陪着M个熊孩子玩一个无聊的游戏:有N个盒子从左到右排成一排,第i个盒子里装着Ai个气球。
SHUXK 要进行Q次操作,每次从某一个盒子里拿出一个没被踩爆的气球,然后熊孩子们就会立刻把它踩爆。
这M个熊孩子每个人都指定了一个盒子区间[Li, Ri]。 如果某一个时刻,一个熊孩子发现自己选定的盒子区间[Li, Ri]中的所
有气球都已经被踩爆了,他就会非常高兴(显然之后他一直会很高兴)。
为了不辜负将自己的任务强行塞给 SHUXK 的那个人的期望, SHUXK 想向你询问:
他每次操作过后会有多少个熊孩子很高兴。
第一行包含两个正整数N和M,分别表示盒子和熊孩子的个数。
第二行包含N个正整数Ai( 1 < = Ai < = 10^5),表示每个盒子里气球的数量。
以下M行每行包含两个正整数Li, Ri( 1 < = Li < = Ri < = N),分别表示每一个熊孩子指定的区间。
以下一行包含一个正整数Q,表示 SHUXK 操作的次数。
以下Q行每行包含一个正整数X,表示这次操作是从第X个盒子里拿气球。为
了体现在线,我们对输入的X进行了加密。
假设输入的正整数是x‘,那么真正的X = (x‘ + Lastans − 1)Mod N + 1。其
中Lastans为上一次询问的答案。对于第一个询问, Lastans = 0。
输入数据保证1 < = x‘ < = 10^9, 且第X个盒子中有尚未被踩爆的气球。
N < = 10^5 ,M < = 10^5 ?,Q < = 10^5
包含Q行,每行输出一个整数,表示 SHUXK 一次操作后询问的
答案。答案的顺序应与输入数据的顺序保持一致。
既然我们必须每一次去查询当前区间全部为0的区间个数有多少个,我们不妨计算每次新增了多少个区间,考虑每一次把一个盒子变为空,就会合并前后两个全0区间,那么当前的点对新增答案的贡献就是左端点位于[包含当前点的极大0区间左端点,当前点]并且右端点位于[当前点,包含当前点的极大子区间的右端点]的区间个数,那么我们维护一个可持久化线段树,每个可持久化线段树中维护右端点位于某个点的区间个数,然后每一次二分合法左端点区间的可持久化线段树,然后查询合法右端点区间个数,更新答案就好了...
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
//by NeighThorn
using namespace std;
const int maxn=100000+5,maxm=7000000+5;
int n,m,Q,ans,tot,a[maxn],ls[maxm],rs[maxm],sum[maxm],root[maxn],fapre[maxn],fanxt[maxn];
struct M{
int l,r;
friend bool operator < (M a,M b){
if(a.l!=b.l)
return a.l<b.l;
return a.r<b.r;
}
}q[maxn];
inline int findpre(int x){
return fapre[x]==x?x:fapre[x]=findpre(fapre[x]);
}
inline int findnxt(int x){
return fanxt[x]==x?x:fanxt[x]=findnxt(fanxt[x]);
}
inline void change(int l,int r,int pos,int x,int &y){
y=++tot;sum[y]=sum[x]+1;
if(l==r)
return;
int mid=(l+r)>>1;ls[y]=ls[x];rs[y]=rs[x];
if(pos<=mid)
change(l,mid,pos,ls[x],ls[y]);
else
change(mid+1,r,pos,rs[x],rs[y]);
}
inline int query(int l,int r,int L,int R,int x,int y){
if(l==L&&r==R)
return sum[y]-sum[x];
int mid=(l+r)>>1;
if(R<=mid)
return query(l,mid,L,R,ls[x],ls[y]);
else if(L>mid)
return query(mid+1,r,L,R,rs[x],rs[y]);
else
return query(l,mid,L,mid,ls[x],ls[y])+query(mid+1,r,mid+1,R,rs[x],rs[y]);
}
inline int findgreater(int x){
int l=1,r=m,res=n+1;
while(l<=r){
int mid=(l+r)>>1;
if(q[mid].l>=x)
res=mid,r=mid-1;
else
l=mid+1;
}
return res;
}
inline int findlower(int x){
int l=1,r=m,res=0;
while(l<=r){
int mid=(l+r)>>1;
if(q[mid].l<=x)
res=mid,l=mid+1;
else
r=mid-1;
}
return res;
}
signed main(void){
scanf("%d%d",&n,&m);fanxt[n+1]=n+1;
for(int i=1;i<=n;i++)
scanf("%d",&a[i]),fapre[i]=i,fanxt[i]=i;
for(int i=1;i<=m;i++)
scanf("%d%d",&q[i].l,&q[i].r);
sort(q+1,q+m+1);scanf("%d",&Q);ans=0;
for(int i=1;i<=m;i++)
change(1,n,q[i].r,root[i-1],root[i]);
for(int i=1,pos,x,y,l,r;i<=Q;i++){
scanf("%d",&pos);pos=(pos+ans-1)%n+1;
a[pos]--;
if(a[pos]!=0) {printf("%d\n",ans);continue;}
x=findpre(pos-1)+1;
y=findnxt(pos+1)-1;
fapre[pos]=pos-1;
fanxt[pos]=pos+1;
l=findgreater(x);
r=findlower(pos);
if(l<=r)
ans+=query(1,n,pos,y,root[l-1],root[r]);
printf("%d\n",ans);
}
return 0;
}