BZOJ 4631: 踩气球

Posted NeighThorn

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了BZOJ 4631: 踩气球相关的知识,希望对你有一定的参考价值。

4631: 踩气球

Time Limit: 10 Sec  Memory Limit: 256 MB
Submit: 275  Solved: 140
[Submit][Status][Discuss]

Description

六一儿童节到了, SHUXK 被迫陪着M个熊孩子玩一个无聊的游戏:有N个盒子从左到右排成一排,第i个盒子里装着Ai个气球。
SHUXK 要进行Q次操作,每次从某一个盒子里拿出一个没被踩爆的气球,然后熊孩子们就会立刻把它踩爆。
这M个熊孩子每个人都指定了一个盒子区间[Li, Ri]。 如果某一个时刻,一个熊孩子发现自己选定的盒子区间[Li, Ri]中的所
有气球都已经被踩爆了,他就会非常高兴(显然之后他一直会很高兴)。
为了不辜负将自己的任务强行塞给 SHUXK 的那个人的期望, SHUXK 想向你询问: 
他每次操作过后会有多少个熊孩子很高兴。

Input

第一行包含两个正整数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

Output

包含Q行,每行输出一个整数,表示 SHUXK 一次操作后询问的
答案。答案的顺序应与输入数据的顺序保持一致。

Sample Input

5 3
1 1 1 1 1
5 5
2 2
1 3
5
4
2
5
2
3

Sample Output

0
1
1
2
3
【样例说明】
实际上每次操作的盒子是: 4 2 1 3 5
在第二次操作后,第二个熊孩子会高兴 (区间[2,2]中的气球已经全部被踩爆)。
在第四次操作后,第三个熊孩子会高兴(区间[1,3]中的气球已经全部被踩爆)。
在第五次操作后,第一个熊孩子会高兴(区间[5,5]中的气球已经全部被踩爆)。

HINT

Source

分析:

我一开始看成离线的了...然后这不是水水嘛,就开始写,发现是在线QwQ~~~

既然我们必须每一次去查询当前区间全部为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;
}

  


By NeighThorn

以上是关于BZOJ 4631: 踩气球的主要内容,如果未能解决你的问题,请参考以下文章

BZOJ 4631: 踩气球

bzoj 4631: 踩气球 线段树合并

BZOJ4631踩气球 链表+线段树+堆

bzoj4631 踩气球

noj算法 踩气球 回溯法

P4215 踩气球