题目描述

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

输入

输入的第一行为两个正整数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

输出

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

样例输入

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

样例输出

NAIVE!ORZzyz.
NAIVE!ORZzyz.
3


题解

权值线段树套KD-tree

对于外层权值线段树的每个节点,对应着一棵KD-tree,存储权值在外层节点范围内的所有点。

对于每次询问,如果判定有解,则寻找区间右半部分中点的个数,如果大于等于k则在右区间中找,否则在左区间中找,直至l=r得到答案。

嗯,说起来真是容易。然而这样交上去肯定是必T无疑。

究其原因就是在KD-tree上。

一开始把“平衡”KD-tree的时间复杂度当成$O(\log n)$的了,以为稳过,结果卡得很死。

由于KD-tree不能旋转什么的,所以自然不能保证平衡。

那么我们能做的只有对KD-tree进行重构,但是蒟蒻又不会部分重构(子树大小超过某比例时重构,替罪羊树的操作),所以只能固定点数重构。

然而重构又出了各种各样的问题 = =,改了以后交上去还是TLE。

实在没办法,要了份数据,发现第6、7个点卡不重构的KD-tree,第8、9、10个点数据范围非常大,这导致无论怎么改重构时间都无法通过。

最后只能针对数据来解决问题了(逃),m=50000时是前7个点,m=100000时是后3个点,分情况就好了,最后还是勉强跑过了。

说实话真正考试如果出这种题的话80分真的是满足了。

#include <cstdio>
#include <cstring>
#include <algorithm>
#define N 3000010
using namespace std;
const int M = 1000000000;
int m , R , ls[N] , rs[N] , root[N] , ts[N] , tot , d , num , x1 , y1 , x2 , y2;
int sta[N] , top;
struct data
{
int p[2] , mx[2] , mn[2] , si , c[2];
bool operator<(const data a)const {return p[d] == a.p[d] ? p[d ^ 1] < a.p[d ^ 1] : p[d] < a.p[d];}
}a[N];
bool cmp(int x , int y)
{
return a[x].p[d] == a[y].p[d] ? a[x].p[d ^ 1] < a[y].p[d ^ 1] : a[x].p[d] < a[y].p[d];
}
void pushup(int k)
{
int l = a[k].c[0] , r = a[k].c[1];
a[k].mx[0] = max(a[k].p[0] , max(a[l].mx[0] , a[r].mx[0]));
a[k].mx[1] = max(a[k].p[1] , max(a[l].mx[1] , a[r].mx[1]));
a[k].mn[0] = min(a[k].p[0] , min(a[l].mn[0] , a[r].mn[0]));
a[k].mn[1] = min(a[k].p[1] , min(a[l].mn[1] , a[r].mn[1]));
a[k].si = a[l].si + a[r].si + 1;
}
void insert(int &k)
{
if(!k) k = ++num , a[k].p[0] = x1 , a[k].p[1] = y1;
else if((d == 0 && (x1 == a[k].p[0] ? y1 < a[k].p[1] : x1 < a[k].p[0])) || (d == 1 && (y1 == a[k].p[1] ? x1 < a[k].p[0] : y1 < a[k].p[1]))) d ^= 1 , insert(a[k].c[0]);
else insert(a[k].c[1]);
pushup(k);
}
int query(int k)
{
if(!k || x1 > a[k].mx[0] || y1 > a[k].mx[1] || x2 < a[k].mn[0] || y2 < a[k].mn[1]) return 0;
if(x1 <= a[k].mn[0] && y1 <= a[k].mn[1] && x2 >= a[k].mx[0] && y2 >= a[k].mx[1]) return a[k].si;
int ans = (a[k].p[0] >= x1 && a[k].p[1] >= y1 && a[k].p[0] <= x2 && a[k].p[1] <= y2);
ans += query(a[k].c[0]);
ans += query(a[k].c[1]);
return ans;
}
int solve(int k , int l , int r , int x)
{
if(l == r) return l;
int mid = (l + r) >> 1 , sum = query(root[rs[x]]);
if(sum >= k) return solve(k , mid + 1 , r , rs[x]);
else return solve(k - sum , l , mid , ls[x]);
}
int build(int l , int r , int now)
{
if(l > r) return 0;
int mid = (l + r) >> 1 , pos;
d = now , nth_element(sta + l , sta + mid , sta + r + 1 , cmp) , pos = sta[mid];
a[pos].c[0] = build(l , mid - 1 , now ^ 1);
a[pos].c[1] = build(mid + 1 , r , now ^ 1);
pushup(pos);
return pos;
}
void dfs(int k)
{
if(!k) return;
sta[++top] = k;
dfs(a[k].c[0]) , dfs(a[k].c[1]);
}
void rebuild(int &k)
{
top = 0 , dfs(k);
k = build(1 , top , 0);
}
void add(int p , int l , int r , int &x)
{
if(!x) x = ++tot;
d = 0 , insert(root[x]) , ts[x] ++ ;
if(m == 50000 && ts[x] >= 2000) rebuild(root[x]) , ts[x] = 0;
if(l == r) return;
int mid = (l + r) >> 1;
if(p <= mid) add(p , l , mid , ls[x]);
else add(p , mid + 1 , r , rs[x]);
}
int main()
{
a[0].mx[0] = a[0].mx[1] = -M , a[0].mn[0] = a[0].mn[1] = M;
int ans = 0 , opt , v , i;
scanf("%*d%d" , &m);
for(i = 1 ; i <= m ; i ++ )
{
scanf("%d" , &opt);
if(opt == 1)
{
scanf("%d%d%d" , &x1 , &y1 , &v);
x1 ^= ans , y1 ^= ans , v ^= ans;
add(v , 1 , M , R);
}
else
{
scanf("%d%d%d%d%d" , &x1 , &y1 , &x2 , &y2 , &v);
x1 ^= ans , y1 ^= ans , x2 ^= ans , y2 ^= ans , v ^= ans;
if(query(root[R]) < v) puts("NAIVE!ORZzyz.") , ans = 0;
else printf("%d\n" , ans = solve(v , 1 , M , R));
}
}
return 0;
}

17.12.18 UPD

写了一下替罪羊式重构的写法,好像也不是很难嘛。。。

在KD-tree插入过程中直接判断一个点是否失去平衡,是的话就直接重构即可。方法和替罪羊树类似,可以参考替罪羊树例题 【bzoj3065】带插入区间k小值

并且稍微优化了一下代码的细节,现在差不多可读了。。。

时间复杂度终于是稳定的 $O(n\sqrt n\log n)$ 了,终于不用针对数据啦。

#include <cstdio>
#include <algorithm>
#define N 100010
#define M 3000010
using namespace std;
struct seg
{
int ls , rs , rb;
}a[M];
struct kdt
{
int p[2] , mx[2] , mn[2] , ls , rs , si;
}b[M];
int A[2] , B[2] , d , root , id[N] , ta , tb , tot;
inline bool tc(int *a , int *b)
{
return a[d] == b[d] ? a[d ^ 1] < b[d ^ 1] : a[d] < b[d];
}
bool cmp(int x , int y)
{
return tc(b[x].p , b[y].p);
}
inline void pushup(int x)
{
int l = b[x].ls , r = b[x].rs;
b[x].mx[0] = max(b[x].p[0] , max(b[l].mx[0] , b[r].mx[0]));
b[x].mx[1] = max(b[x].p[1] , max(b[l].mx[1] , b[r].mx[1]));
b[x].mn[0] = min(b[x].p[0] , min(b[l].mn[0] , b[r].mn[0]));
b[x].mn[1] = min(b[x].p[1] , min(b[l].mn[1] , b[r].mn[1]));
b[x].si = b[l].si + b[r].si + 1;
}
int build(int l , int r , int now)
{
if(l > r) return 0;
int mid = (l + r) >> 1;
d = now , nth_element(id + l , id + mid , id + r + 1 , cmp);
b[id[mid]].ls = build(l , mid - 1 , now ^ 1);
b[id[mid]].rs = build(mid + 1 , r , now ^ 1);
pushup(id[mid]);
return id[mid];
}
void dfs(int k)
{
if(k) dfs(b[k].ls) , id[++tot] = k , dfs(b[k].rs);
}
void insert(int &k , int now , int flag)
{
if(!k)
{
k = ++tb , b[k].p[0] = A[0] , b[k].p[1] = A[1] , pushup(k);
return;
}
bool tag;
d = now;
if(tc(A , b[k].p)) tag = ((b[b[k].ls].si + 1) * 4 > (b[k].si + 1) * 3) , insert(b[k].ls , now ^ 1 , tag | flag);
else tag = ((b[b[k].rs].si + 1) * 4 > (b[k].si + 1) * 3) , insert(b[k].rs , now ^ 1 , tag | flag);
pushup(k);
if(tag && !flag) tot = 0 , dfs(k) , k = build(1 , tot , now);
}
int query(int k)
{
if(!k || b[k].mx[0] < A[0] || b[k].mx[1] < A[1] || b[k].mn[0] > B[0] || b[k].mn[1] > B[1]) return 0;
if(b[k].mn[0] >= A[0] && b[k].mn[1] >= A[1] && b[k].mx[0] <= B[0] && b[k].mx[1] <= B[1]) return b[k].si;
return (b[k].p[0] >= A[0] && b[k].p[1] >= A[1] && b[k].p[0] <= B[0] && b[k].p[1] <= B[1]) + query(b[k].ls) + query(b[k].rs);
}
void modify(int p , int l , int r , int &x)
{
if(!x) x = ++ta;
insert(a[x].rb , 0 , 0);
if(l == r) return;
int mid = (l + r) >> 1;
if(p <= mid) modify(p , l , mid , a[x].ls);
else modify(p , mid + 1 , r , a[x].rs);
}
int solve(int k , int l , int r , int x)
{
if(l == r) return l;
int mid = (l + r) >> 1 , t = query(a[a[x].rs].rb);
if(k <= t) return solve(k , mid + 1 , r , a[x].rs);
else return solve(k - t , l , mid , a[x].ls);
}
int main()
{
b[0].mx[0] = b[0].mx[1] = -1 << 30 , b[0].mn[0] = b[0].mn[1] = 1 << 30;
int m , opt , x , last = 0;
scanf("%*d%d" , &m);
while(m -- )
{
scanf("%d" , &opt);
if(opt == 1) scanf("%d%d%d" , &A[0] , &A[1] , &x) , A[0] ^= last , A[1] ^= last , x ^= last , modify(x , 0 , 1000000000 , root);
else
{
scanf("%d%d%d%d%d" , &A[0] , &A[1] , &B[0] , &B[1] , &x), A[0] ^= last , A[1] ^= last , B[0] ^= last , B[1] ^= last , x ^= last;
if(!(last = solve(x , 0 , 1000000000 , root))) puts("NAIVE!ORZzyz.");
else printf("%d\n" , last);
}
}
return 0;
}

【bzoj4605】崂山白花蛇草水 权值线段树套KD-tree的更多相关文章

  1. bzoj4605: 崂山白花蛇草水 权值线段树套KDtree

    bzoj4605: 崂山白花蛇草水 链接 bzoj loj 思路 强制在线,那就权值线段树套KDtree好了,没啥好讲的. KDtree要加平衡因子来重构.另外,那水真难喝. 错误 树套树一边写过了, ...

  2. 崂山白花蛇草水 权值线段树套KDtree

    Description 神犇Aleph在SDOI Round2前立了一个flag:如果进了省队,就现场直播喝崂山白花蛇草水.凭借着神犇Aleph的实 力,他轻松地进了山东省省队,现在便是他履行诺言的时 ...

  3. 【BZOJ4605】崂山白花蛇草水 权值线段树+kd-tree

    [BZOJ4605]崂山白花蛇草水 Description 神犇Aleph在SDOI Round2前立了一个flag:如果进了省队,就现场直播喝崂山白花蛇草水.凭借着神犇Aleph的实力,他轻松地进了 ...

  4. 洛谷P4848 崂山白花蛇草水 权值线段树+KDtree

    题目描述 神犇 \(Aleph\) 在 \(SDOI\ Round2\) 前立了一个 \(flag\):如果进了省队,就现场直播喝崂山白花蛇草水.凭借着神犇 \(Aleph\) 的实力,他轻松地进了山 ...

  5. [bzoj3196][Tyvj1730]二逼平衡树_树套树_位置线段树套非旋转Treap/树状数组套主席树/权值线段树套位置线段树

    二逼平衡树 bzoj-3196 Tyvj-1730 题目大意:请写出一个维护序列的数据结构支持:查询给定权值排名:查询区间k小值:单点修改:查询区间内定值前驱:查询区间内定值后继. 注释:$1\le ...

  6. [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)

    [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...

  7. BZOJ 3110 ZJOI 2013 K大数查询 树套树(权值线段树套区间线段树)

    题目大意:有一些位置.这些位置上能够放若干个数字. 如今有两种操作. 1.在区间l到r上加入一个数字x 2.求出l到r上的第k大的数字是什么 思路:这样的题一看就是树套树,关键是怎么套,怎么写.(话说 ...

  8. BZOJ3110[Zjoi2013]K大数查询——权值线段树套线段树

    题目描述 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数是 ...

  9. 【bzoj3110】[Zjoi2013]K大数查询 权值线段树套区间线段树

    题目描述 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c.如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数 ...

随机推荐

  1. Android笔记--Bitmap(三) 针对不用Android版本的位图管理

    Bitmap(三) | Android不同版本的相应操作 在不同的Android版本中.位图的存储方式是不同的. 1.小于等于 Android 2.2 (API level 8) 垃圾收集器回收内存时 ...

  2. android Random的使用

    一.Random 此类的实例用于生成伪随机数流.此类使用 48 位的种子,使用线性同余公式 (linear congruential form) 对其进行了修改. 如果用相同的种子创建两个 Rando ...

  3. [windows]桌面中添加我的电脑,我的文档和网上邻居图标

    xp系统: 操作步骤:桌面任意位置--〉右键--〉属性--〉桌面选项卡--〉自定义桌面--〉常规:勾选相关图标确定即可. win7系统: 操作步骤:桌面任意位置--〉右键--〉个性化--〉(右侧)更改 ...

  4. Mysql如何为表字段添加索引???

    1.添加PRIMARY KEY(主键索引): ALTER TABLE `table_name` ADD PRIMARY KEY ( `column` ) 2.添加UNIQUE(唯一索引) : ALTE ...

  5. SQLServer查询死锁

    --查询死锁 select request_session_id spid, OBJECT_NAME(resource_associated_entity_id) tableName from sys ...

  6. PBI DAX 中GroupBy

    平时工作中经常会遇到Group By 的情形,用sql 写group by 很容易,在PBI中可以这样写: SUMMARIZE(表名,GroupBy Key ,"聚合列命名",DI ...

  7. 11gR2新特性---Gpnp守护进程

    在这篇文章中,我们会对11gR2 新的守护进程(资源名称ora.gpnpd)进行介绍,其中包含的gpnp的功能,启动顺序和基本的诊断方法. gpnp全称为grid plug and play,该组件的 ...

  8. MINST手写数字识别(二)—— 卷积神经网络(CNN)

    今天我们的主角是keras,其简洁性和易用性简直出乎David 9我的预期.大家都知道keras是在TensorFlow上又包装了一层,向简洁易用的深度学习又迈出了坚实的一步. 所以,今天就来带大家写 ...

  9. VS打包软件部署------ClickOnce应用安装 (各版本.net引导文件安装,再发布文档离线安装下载地址)

    一.1.其他引导程序包:地址  https://msdn.microsoft.com/zh-cn/vstudio/bb898654.aspx 2.离线安装各版本.net下载版(各种语言):https: ...

  10. Bootstrap 响应式表格

    响应式表格 通过把任意的 .table 包在 .table-responsive class 内,您可以让表格水平滚动以适应小型设备(小于 768px).当在大于 768px 宽的大型设备上查看时,您 ...