bzoj 4571 [Scoi2016]美味——主席树

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4571

按位考虑,需要的就是一个区间;比如最高位就是(2^k -x)。

对于不是最高位的位置该怎么考虑?其实之前位置如果能或不能匹配上,也就相当于指定了之前的位上的是0还是1;把是1的位累计进一个变量里,加到区间的边界上就行了!

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=2e5+5,M=1e5+5,K=N*20;
int n,m,mx,a[N],rt[N],tot,sm[K],ls[K],rs[K];
int ans,lj,lo,hi,bin[20];
int rdn()
{
    int ret=0;bool fx=1;char ch=getchar();
    while(ch>'9'||ch<'0'){if(ch=='-')fx=0;ch=getchar();}
    while(ch>='0'&&ch<='9') ret=(ret<<3)+(ret<<1)+ch-'0',ch=getchar();
    return fx?ret:-ret;
}
void build(int &cr,int pre,int l,int r,int v)
{
    cr=++tot; ls[cr]=ls[pre]; rs[cr]=rs[pre];
    sm[cr]=sm[pre]+1;
    if(l==r) return; int mid=l+r>>1;
    if(v<=mid) build(ls[cr],ls[pre],l,mid,v);
    else build(rs[cr],rs[pre],mid+1,r,v);
}
bool query(int cr,int pre,int l,int r,int L,int R)
{
    if(l>=L&&r<=R) return sm[cr]-sm[pre];
    int mid=l+r>>1; bool fg=0;
    if(L<=mid) fg=query(ls[cr],ls[pre],l,mid,L,R);
    if(!fg&&mid<R) fg=query(rs[cr],rs[pre],mid+1,r,L,R);
    return fg;
}
void init()
{
    bin[0]=1;
    for(int i=1;i<=18;i++) bin[i]=bin[i-1]<<1;
}
int main()
{
    n=rdn(); m=rdn(); init();
    for(int i=1;i<=n;i++) a[i]=rdn(),mx=max(mx,a[i]);
    for(int i=1;i<=n;i++)
        build(rt[i],rt[i-1],1,mx,a[i]);
    for(int i=1,b,x,l,r;i<=m;i++)
    {
        b=rdn(); x=rdn(); l=rdn(); r=rdn();
        ans=0;  lj=0;
        for(int j=17;j>=0;j--)
        {
            if(b&bin[j]) {lo=lj; hi=lj+bin[j]-1;}
            else {lo=lj+bin[j]; hi=lj+bin[j+1]-1;}
            hi-=x; lo-=x;
            hi=min(hi,mx); lo=max(lo,0);

            if(lo<=hi&&query(rt[r],rt[l-1],1,mx,lo,hi))
                lj+=(b&bin[j]?0:bin[j]),ans+=bin[j];
            else lj+=(b&bin[j]?bin[j]:0);
        }
        printf("%d\n",ans);
    }
    return 0;
}

 

posted on 2018-09-29 19:55  Narh  阅读(89)  评论(0编辑  收藏  举报

导航