【线段树套KD树】[BZOJ4605]崂山白花蛇草水

题目描述

Description

神犇Aleph在SDOI Round2前立了一个flag:如果进了省队,就现场直播喝崂山白花蛇草水。凭借着神犇Aleph的实
力,他轻松地进了山东省省队,现在便是他履行诺言的时候了。蒟蒻Bob特地为他准备了999,999,999,999,999,999
瓶崂山白花蛇草水,想要灌神犇Aleph。神犇Aleph求(跪着的)蒟蒻Bob不要灌他,由于神犇Aleph是神犇,蒟蒻Bo
b最终答应了他的请求,但蒟蒻Bob决定将计就计,也让神犇Aleph回答一些问题。具体说来,蒟蒻Bob会在一个宽敞
的广场上放置一些崂山白花蛇草水(可视为二维平面上的一些整点),然后询问神犇Aleph在矩形区域(x1, y1), (
x2, y2)(x1≤x2且y1≤y2,包括边界)中,崂山白花蛇草水瓶数第k多的是多少。为了避免麻烦,蒟蒻Bob不会在同
一个位置放置两次或两次以上的崂山白花蛇草水,但蒟蒻Bob想为难一下神犇Aleph,希望他能在每次询问时立刻回
答出答案。神犇Aleph不屑于做这种问题,所以把这个问题交给了你。
Input

输入的第一行为两个正整数N, Q,表示横纵坐标的范围和蒟蒻Bob的操作次数(包括放置次数和询问次数)。
接下来Q行,每行代表蒟蒻Bob的一个操作,操作格式如下:
首先第一个数字type,表示操作种类。type=1表示放置,type=2表示询问。
若type=1,接下来会有三个正整数x, y, v,表示在坐标整点(x, y)放置v瓶崂山白花蛇草水。(1≤x, y≤N, 1≤v≤10^9)
若type=2,接下来会有五个正整数x1, y1, x2, y2, k,表示询问矩形区域(x1, y1), (x2, y2)中,崂山白花蛇草水瓶数第k多的是多少。
(1≤x1≤x2≤N,1≤y1≤y2≤N,1≤k≤Q)
为了体现程序的在线性,你需要将每次读入的数据(除了type值)都异或lastans,其中lastans表示上次询问的答
案。如果上次询问的答案为”NAIVE!ORZzyz.”(见样例输出),则将lastans置为0。初始时的lastans为0。
初始时平面上不存在崂山白花蛇草水。
本题共有12组测试数据。对于所有的数据,N≤500,000。
Q的范围见下表:
测试点1-2 Q=1,000
测试点3-7 Q=50,000
测试点8-12 Q=100,000
Output

对于每个询问(type=2的操作),回答崂山白花蛇草水瓶数第k多的是多少。若不存在第k多的瓶数,
请输出”NAIVE!ORZzyz.”(输出不含双引号)。
Sample Input

10 7

1 1 1 1

1 2 2 3

1 4 1 2

1 3 4 4

2 1 1 4 1 3

2 2 2 3 5 4

2 2 1 4 4 2
Sample Output

NAIVE!ORZzyz.

NAIVE!ORZzyz.

3
HINT

Source

Idea By Aleph, Description & Data cases By jinlifu1999.

分析

本题强制在线,显然是一道数据结构题,维护的是一个二维区间,想到树套树或者K-D树,还要寻找第k大,所以还需要再套一棵树。树套树套树编码难度和常数都很大,所以我们选择K-D树。另一棵树显然有很多种选择,但是显然线段树是我们最佳的选择,因为它的形态不会改变。由于K-D树随着节点的增加会变得不平衡,我们可以使用类似替罪羊树的重建方法对K-D树进行部分重构,如果放外层重建的时空开销会比较大。所以我们外层维护一棵权值线段树,内层维护一棵K-D树即可时间复杂度O(nnlog2n)

代码

#include<cstdio>
#include<algorithm>
#include<queue>
#include<vector>
#define alpha 0.77
#define MAXV 1000000000
#define MAXN 100000
using namespace std;
int n,Q,ans;
void Read(int &x){
    char c;
    while(c=getchar(),c!=EOF)
        if(c>='0'&&c<='9'){
            x=c-'0';
            while(c=getchar(),c>='0'&&c<='9')
                x=x*10+c-'0';
            ungetc(c,stdin);
            return;
        }
}
namespace K_DTree{
int D;
bool rbd;
struct node{
    int d[2],mn[2],mx[2],cnt,sum;
    node *ch[2];
    inline node(){
    }
    inline node(int x,int y):cnt(1){
        d[0]=mn[0]=mx[0]=x,d[1]=mx[1]=mn[1]=y;
    }
    inline bool operator<(const node &b)const{
        return d[D]<b.d[D];
    }
    inline bool operator==(const node &b)const{
        return d[0]==b.d[0]&&d[1]==b.d[1];
    }
}now,tree[MAXN*25+10],*tcnt=tree,*v[MAXN+10],**vcnt,**rbt;
inline bool cmp(node *x,node *y){
    return x->d[D]<y->d[D];
}
inline int Get_sum(node *p){
    return p?p->sum:0;
}
inline bool in(int x1,int y1,int x2,int y2,int X1,int Y1,int X2,int Y2){
    return x1<=X1&&X2<=x2&&y1<=Y1&&Y2<=y2;
}
inline bool out(int x1,int y1,int x2,int y2,int X1,int Y1,int X2,int Y2){
    return x1>X2||x2<X1||y1>Y2||y2<Y1;
}
inline bool balance(node *p){
    return p->sum*alpha>=max(Get_sum(p->ch[0]),Get_sum(p->ch[1]));
}
void update(node *p){
    for(int i=0;i<2;i++){
        p->mx[i]=p->mn[i]=p->d[i];
        if(p->ch[0])
            p->mn[i]=min(p->mn[i],p->ch[0]->mn[i]),p->mx[i]=max(p->mx[i],p->ch[0]->mx[i]);
        if(p->ch[1])
            p->mn[i]=min(p->mn[i],p->ch[1]->mn[i]),p->mx[i]=max(p->mx[i],p->ch[1]->mx[i]);
    }
    p->sum=Get_sum(p->ch[0])+Get_sum(p->ch[1])+p->cnt;
}
void rebuild(node *&p,int l,int r,bool f){
    if(l>r)
        return;
    D=f;
    int mid=(l+r)>>1;
    nth_element(v+l,v+mid,v+r+1,cmp);
    p=v[mid];
    p->ch[0]=p->ch[1]=0;
    rebuild(p->ch[0],l,mid-1,f^1);
    rebuild(p->ch[1],mid+1,r,f^1);
    update(p);
}
void travel(node *p){
    if(!p)
        return;
    *++vcnt=p;
    travel(p->ch[0]);
    travel(p->ch[1]);
}
void insert(node *&p,bool D){
    if(!p){
        p=++tcnt;
        p->cnt=p->sum=0;
        p->ch[0]=p->ch[1]=0;
        p->d[0]=p->mn[0]=p->mx[0]=now.d[0];
        p->d[1]=p->mn[1]=p->mx[1]=now.d[1];
    }
    if(*p==now){
        p->sum++;
        p->cnt++;
        return;
    }
    if(now.d[D]<p->d[D])
        insert(p->ch[0],D^1);
    else
        insert(p->ch[1],D^1);
    update(p);
    if(!balance(p))
        rbt=&p,rbd=D;
}
void rebuild(node *&rt,bool f){
    vcnt=v;
    travel(rt);
    rt=0;
    rebuild(rt,1,vcnt-v,f);
}
inline void insert(node *&rt){
    rbt=0;
    insert(rt,0);
    if(rbt)
        rebuild(*rbt,rbd);
}
int get_sum(node *p,int x1,int y1,int x2,int y2){
    if(!p)
        return 0;
    if(in(x1,y1,x2,y2,p->mn[0],p->mn[1],p->mx[0],p->mx[1]))
        return p->sum;
    if(out(x1,y1,x2,y2,p->mn[0],p->mn[1],p->mx[0],p->mx[1]))
        return 0;
    return get_sum(p->ch[0],x1,y1,x2,y2)+get_sum(p->ch[1],x1,y1,x2,y2)+p->cnt*in(x1,y1,x2,y2,p->d[0],p->d[1],p->d[0],p->d[1]);
}
}
int x[MAXN+10],y[MAXN+10],val[MAXN+10];
namespace SegmentTree{
struct node{
    K_DTree::node *root;
    node *ch[2];
}tree[MAXN*20+10],*tcnt=tree,*nil=tree,*root=nil;
inline void init(){
    nil->ch[0]=nil->ch[1]=nil;
}
void insert(node *&p,int l,int r,int pos){
    if(p==nil){
        p=++tcnt;
        p->ch[0]=p->ch[1]=nil;
    }
    K_DTree::insert(p->root);
    if(l==r)
        return;
    int mid=(l+r)>>1;
    if(pos<=mid)
        insert(p->ch[0],l,mid,pos);
    else
        insert(p->ch[1],mid+1,r,pos);
}
int Query(node *p,int l,int r,int x1,int y1,int x2,int y2,int k){
    if(l==r)
        return l;
    int mid=(l+r)>>1,t;
    if((t=K_DTree::get_sum(p->ch[1]->root,x1,y1,x2,y2))>=k)
        return Query(p->ch[1],mid+1,r,x1,y1,x2,y2,k);
    return Query(p->ch[0],l,mid,x1,y1,x2,y2,k-t);
}
}
using namespace SegmentTree;
int main()
{
    init();
    Read(n),Read(Q);
    int x,y,v,x1,y1,p;
    while(Q--){
        Read(p);
        if(p==1){
            Read(x),Read(y),Read(v);
            x^=ans,y^=ans,v^=ans;
            K_DTree::now=K_DTree::node(x,y);
            insert(root,1,MAXV,v);
        }
        else{
            Read(x),Read(y),Read(x1),Read(y1),Read(v);
            x^=ans,y^=ans,x1^=ans,y1^=ans,v^=ans;
            if(K_DTree::get_sum(root->root,x,y,x1,y1)<v)
                puts("NAIVE!ORZzyz."),ans=0;
            else
                printf("%d\n",ans=Query(root,1,MAXV,x,y,x1,y1,v));
        }
    }
}

转载于:https://www.cnblogs.com/outerform/p/5921826.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值