BZOJ3489: A simple rmq problem kdtree

题意:给出一个长度为N的序列,M个询问,每次询问[L,R]中只出现过一次的最大数,不存在输出0,强制在线
N<=100000 M<=200000
卡时神器kdtree暴力水过。
用set预处理出每个元素i前面一个和它相等的元素pre[i]和后面一个和它相等的元素nex[i],则查询变为查询l<=pos<=r,pre < l,nex > r的元素中最大的,用三维kdtree暴力就可以了。
读入优化加上更慢是什么情况。。。

#include<cstdio>
#include<algorithm>
#include<ext/pb_ds/assoc_container.hpp>
#define gm 100005
#ifdef ONLINE_JUDGE
#define null_type null_mapped_type
#endif
using namespace __gnu_pbds;
int cmp;
struct pnt
{
    int p[3];
    int& operator [] (size_t x) {return p[x];}
    const int& operator [] (size_t x) const {return p[x];}
}p[gm];
inline void min(int& a,int b)
{
    if(b<a) a=b;
}
inline void max(int& a,int b)
{
    if(a<b) a=b;
}
struct node
{
    node *l,*r;
    pnt p,mn,mx;
    int v,sv;
    node(const pnt &p,size_t v):l(),r(),p(p),mn(p),mx(p),v(v),sv(v){}
    inline void up()
    {
        if(l)
        {
            min(mn[0],l->mn[0]),min(mn[1],l->mn[1]),min(mn[2],l->mn[2]);
            max(mx[0],l->mx[0]),max(mx[1],l->mx[1]),max(mx[2],l->mx[2]);
            max(sv,l->sv);
        }
        if(r)
        {
            min(mn[0],r->mn[0]),min(mn[1],r->mn[1]),min(mn[2],r->mn[2]);
            max(mx[0],r->mx[0]),max(mx[1],r->mx[1]),max(mx[2],r->mx[2]);
            max(sv,r->sv);
        }
    }
    static int x1,x2,y,z;
    int getmax()
    {
        if(!this) return 0;
        if(x1<=mn[0]&&mx[0]<=x2&&mx[1]<y&&mn[2]>z) return sv;
        if(mx[0]<x1||mn[0]>x2||mn[1]>=y||mx[2]<=z) return 0;
        int res=0;
        if(x1<=p[0]&&p[0]<=x2&&p[1]<y&&p[2]>z) res=v;
        max(res,l->getmax());
        max(res,r->getmax());
        return res;
    }
    inline void* operator new (size_t);
};
int node::x1,node::x2,node::y,node::z;
char pool[gm*sizeof(node)];
inline void* node::operator new(size_t)
{
    static int t=-1;
    static node* s=(node*)pool;
    return s+ ++t;
}
inline bool cmp_fn(int a,int b)
{
    return p[a][cmp]<p[b][cmp];
}
int a[gm],ar[gm];
node* build(int* l,int *r,int k)
{
    if(l==r) return 0;
    int* mid=l+(r-l>>1);
    cmp=k;
    std::nth_element(l,mid,r,cmp_fn);
    node *kre=new node(p[*mid],a[p[*mid][0]]);
    kre->l=build(l,mid,k==2?0:k+1);
    kre->r=build(mid+1,r,k==2?0:k+1);
    kre->up();
    return kre;
}
struct kdtree
{
    node *rt;
    kdtree(int* ar,int len):rt(build(ar,ar+len,0)){}
    inline int maxnum(int l,int r)
    {
        node::x1=node::y=l,node::x2=node::z=r;
        return rt->getmax();
    }
};
int n,m,last=0;
typedef std::pair<int,int> pr;
typedef tree<pr,null_type> set;
set s;
set::iterator pos[gm];
const size_t str=1<<20;
struct Istream
{
    char buf[str],*s,*t;
    Istream():buf(),s(),t(){}
    inline char get()
    {
        if(s==t) t=buf+fread(s=buf,1,str-1,stdin);
        return *s++;
    }
    inline Istream& operator >> (int &x)
    {
        register char c;
        x=0;
        do c=get(); while(c<'0'||c>'9');
        while(c>='0'&&c<='9') x=x*10+c-'0',c=get();
        return *this;
    }
}cin;
struct Ostream
{
    char buf[str],*s,*t;
    Ostream():buf(),s(buf),t(buf+str-1){}
    ~Ostream(){fwrite(buf,1,s-buf,stdout);}
    inline void put(char c)
    {
        if(s==t) fwrite(s=buf,1,str-1,stdout);
        *s++=c;
    }
    inline Ostream& operator << (const int &x)
    {
        static int a[15],t;
        if(!x) return put('0'),put('\n'),*this;
        a[t=1]=x;
        while(a[t]) a[t+1]=a[t]/10,a[t++]%=10;
        while(--t) put(a[t]+'0');
        return put('\n'),*this;
    }
}cout;
int main()
{
    cin>>n>>m;
    s.insert(pr(0,0)),s.insert(pr(n+1,n+1));
    for(int i=1;i<=n;++i)
    {
        cin>>a[i];
        pos[i]=s.insert(pr(a[i],i)).first;
        p[i][0]=ar[i]=i;
    }
    for(int i=1;i<=n;++i)
    {
        set::iterator x=pos[i],y=pos[i];
        --x,++y;
        p[i][1]=(a[i]==x->first)?x->second:0;
        p[i][2]=(a[i]==y->first)?y->second:n+1;
    }
    kdtree tr(ar+1,n);
    for(int i=1;i<=m;++i)
    {
        int x,y;
        cin>>x>>y;
        x=(x+last)%n+1,y=(y+last)%n+1;
        if(x>y){int z=x;x=y;y=z;}
        cout<<(last=tr.maxnum(x,y));
    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值