zoukankan      html  css  js  c++  java
  • 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
    ",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
    ",ans);
    	}
    	return 0;
    }
    

      


    By NeighThorn

  • 相关阅读:
    define vs const vs enum
    解决Ubuntu 14.04 LTS 浏览网页速度慢的问题
    C语言两种产生矩阵的方法
    GTK 添加图标
    Unix Socket 端口 reuse
    Linux GTK Hello,World
    插件使用记录
    原型链和new
    each函数循环数据表示列举,列举循环的时候添加dom的方法
    字体圆润属性的使用-webkit-font-smoothing: antialiased
  • 原文地址:https://www.cnblogs.com/neighthorn/p/6539106.html
Copyright © 2011-2022 走看看