题面

传送门

KDTree

大概就是一个分割\(k\)维空间的数据结构,二叉树

建立:每层选取一维为关键字,把中间的点拿出来,递归左右,有个\(STL\)函数nth_element可以用一下

维护:维护当前这个点的子树的每一维的最大值和最小值,相当于维护了个高维矩形

查询:直接遍历一棵树是\(O(n)\)的,利用一些独特的性质可以剪枝,因题而异

奇技淫巧:

  1. 把坐标绕原点转\(\alpha\)度
  2. 定期重构或者像替罪羊树一样,利用平衡因子判断是否需重构
  3. 每次查询到某一层优先选取答案可能最优的一个儿子先递归
  4. 等等

Sol

\(KDTree\)模板题

利用替罪羊树的思想重构

# include <bits/stdc++.h>
# define IL inline
# define RG register
# define Fill(a, b) memset(a, b, sizeof(a))
using namespace std;
typedef long long ll; IL int Input(){
RG int x = 0, z = 1; RG char c = getchar();
for(; c < '0' || c > '9'; c = getchar()) z = c == '-' ? -1 : 1;
for(; c >= '0' && c <= '9'; c = getchar()) x = (x << 1) + (x << 3) + (c ^ 48);
return x * z;
} const int maxn(1e6 + 5);
const int inf(2e9);
const double alpha(0.75); int n, op, rt, q, ans, p[maxn], tot, num, cnt; struct Point{
int d[2]; IL int operator <(RG Point a) const{
return d[op] < a.d[op];
}
} a[maxn]; struct KDTree{
int ch[2], d[2], size;
int mn[2], mx[2];
} tr[maxn]; IL void Chkmax(RG int &x, RG int y){
if(y > x) x = y;
} IL void Chkmin(RG int &x, RG int y){
if(y < x) x = y;
} IL int NewNode(){
return tot ? p[tot--] : ++cnt;
} IL void Update(RG int x){
RG int ls = tr[x].ch[0], rs = tr[x].ch[1];
tr[x].mx[0] = tr[x].mn[0] = tr[x].d[0];
tr[x].mx[1] = tr[x].mn[1] = tr[x].d[1];
tr[x].size = tr[ls].size + tr[rs].size + 1;
if(ls){
Chkmin(tr[x].mn[0], tr[ls].mn[0]), Chkmin(tr[x].mn[1], tr[ls].mn[1]);
Chkmax(tr[x].mx[0], tr[ls].mx[0]), Chkmax(tr[x].mx[1], tr[ls].mx[1]);
}
if(rs){
Chkmin(tr[x].mn[0], tr[rs].mn[0]), Chkmin(tr[x].mn[1], tr[rs].mn[1]);
Chkmax(tr[x].mx[0], tr[rs].mx[0]), Chkmax(tr[x].mx[1], tr[rs].mx[1]);
}
} IL int Build(RG int l, RG int r, RG int nop){
op = nop;
RG int x = (l + r) >> 1, nw = NewNode();
nth_element(a + l, a + x, a + r + 1);
tr[nw].d[0] = a[x].d[0], tr[nw].d[1] = a[x].d[1];
if(l < x) tr[nw].ch[0] = Build(l, x - 1, nop ^ 1);
if(x < r) tr[nw].ch[1] = Build(x + 1, r, nop ^ 1);
Update(nw);
return nw;
} IL int Check(RG int x){
return alpha * tr[x].size < tr[tr[x].ch[0]].size || alpha * tr[x].size < tr[tr[x].ch[1]].size;
} IL void ReCycle(RG int x){
if(!x) return;
ReCycle(tr[x].ch[0]), tr[x].ch[0] = 0;
p[++tot] = x, a[++num].d[0] = tr[x].d[0], a[num].d[1] = tr[x].d[1];
ReCycle(tr[x].ch[1]), tr[x].ch[1] = 0;
} IL int ReBuild(RG int x, RG int nop){
num = 0, ReCycle(x);
return Build(1, num, nop);
} IL void Insert(RG int &x, RG int nop, RG Point np){
if(!x){
x = NewNode();
tr[x].d[0] = np.d[0], tr[x].d[1] = np.d[1];
tr[x].mx[0] = tr[x].mn[0] = tr[x].d[0];
tr[x].mx[1] = tr[x].mn[1] = tr[x].d[1];
return;
}
if(np.d[nop] < tr[x].d[nop]) Insert(tr[x].ch[0], nop ^ 1, np);
else Insert(tr[x].ch[1], nop ^ 1, np);
Update(x);
if(Check(x)) x = ReBuild(x, nop);
} IL int Calc(RG int x, RG Point np){
RG int d = max(np.d[0] - tr[x].mx[0], 0) + max(tr[x].mn[0] - np.d[0], 0);
return d + max(tr[x].mn[1] - np.d[1], 0) + max(np.d[1] - tr[x].mx[1], 0);
} IL void Query(RG int x, RG Point np){
RG int d = abs(tr[x].d[0] - np.d[0]) + abs(tr[x].d[1] - np.d[1]), d1 = inf, d2 = inf;
Chkmin(ans, d);
if(tr[x].ch[0]) d1 = Calc(tr[x].ch[0], np);
if(tr[x].ch[1]) d2 = Calc(tr[x].ch[1], np);
RG int c = 0;
if(d1 > d2) swap(d1, d2), c ^= 1;
if(d1 < ans) Query(tr[x].ch[c], np);
if(d2 < ans) Query(tr[x].ch[!c], np);
} int main(){
n = Input(), q = Input();
for(RG int i = 1; i <= n; ++i) a[i].d[0] = Input(), a[i].d[1] = Input();
rt = Build(1, n, 0);
for(RG int i = 1; i <= q; ++i){
RG int t = Input(), x = Input(), y = Input();
if(t == 1) Insert(rt, 0, (Point){x, y});
else{
ans = inf;
Query(rt, (Point){x, y});
printf("%d\n", ans);
}
}
return 0;
}

KDTree(Bzoj2648: SJY摆棋子)的更多相关文章

  1. BZOJ2648: SJY摆棋子&&2716: [Violet 3]天使玩偶

    BZOJ2648: SJY摆棋子 BZOJ2716: [Violet 3]天使玩偶 BZOJ氪金无极限... 其实这两道是同一题. 附上2648的题面: Description 这天,SJY显得无聊. ...

  2. [BZOJ2648] SJY摆棋子 kd-tree

    2648: SJY摆棋子 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 5421  Solved: 1910[Submit][Status][Disc ...

  3. BZOJ2648 SJY摆棋子(KD-Tree)

    板子题. #include<iostream> #include<cstdio> #include<cmath> #include<cstdlib> # ...

  4. [bzoj2648]SJY摆棋子(带插入kd-tree)

    解题关键:带插入kdtree模板题. #include<iostream> #include<cstdio> #include<cstring> #include& ...

  5. 【kd-tree】bzoj2648 SJY摆棋子

    #include<cstdio> #include<cmath> #include<algorithm> using namespace std; #define ...

  6. 2019.01.14 bzoj2648: SJY摆棋子(kd-tree)

    传送门 kd−treekd-treekd−tree模板题. 题意简述:支持在平面上插入一个点,求对于一个点的最近点对. 思路:cdqcdqcdq是一种很不错的分治方法 只是好像码量有点窒息 所以我用了 ...

  7. Bzoj2648 SJY摆棋子

    Time Limit: 20 Sec  Memory Limit: 128 MB Submit: 3128  Solved: 1067 Description 这天,SJY显得无聊.在家自己玩.在一个 ...

  8. luogu4169 [Violet]天使玩偶/SJY摆棋子 / bzoj2648 SJY摆棋子 k-d tree

    k-d tree + 重构的思想,就能卡过luogu和bzoj啦orz #include <algorithm> #include <iostream> #include &l ...

  9. 【BZOJ2648】SJY摆棋子(KD-Tree)

    [BZOJ2648]SJY摆棋子(KD-Tree) 题面 BZOJ Description 这天,SJY显得无聊.在家自己玩.在一个棋盘上,有N个黑色棋子.他每次要么放到棋盘上一个黑色棋子,要么放上一 ...

随机推荐

  1. mysql 与sqlser group by

    mysql select * ,count(1)  from simccbillm18 group by MonthNum; SqlSer select  col1,col2 from table g ...

  2. Https 客户端与服务器交互过程梳理

    本文试图以通俗易通的方式介绍Https的工作原理,不纠结具体的术语,不考证严格的流程.我相信弄懂了原理之后,到了具体操作和实现的时候,方向就不会错,然后条条大路通罗马.阅读文本需要提前大致了解对称加密 ...

  3. python连接,操作 InfluxDB

    准备工作 启动服务器 执行如下命令: service influxdb start 示例如下: [root@localhost ~]# service influxdb start Starting ...

  4. java 实用Util汇总

    1.Util介绍 Util是工具的意思,一般来说,常常用来描述和业务逻辑没有关系的数据处理. Util一般要和私有方法对比:私有方法一般来说是只是在特地场景下使用的,私有方法越多,代码结构越乱.常见的 ...

  5. pycharm+gitee

    Git操作 前言: 由于各种原因,很多时候我们写代码的电脑并不会随身携带,所以有的时候突发灵感想继续写代码就变得难以实现.相信大部分同学对此都有了解,那就通过代码托管平台来管理.原本想用GitHub来 ...

  6. golang (4) golang 操作mongdb

    1. 数据按照时间聚合操作 1.1 正常的数据结构 { "_id" : ObjectId("5cac8d7b1202708adf5d4b64"), " ...

  7. python爬虫常用之Scrapy 中间件

    一.概述 1.中间件的作用 在scrapy运行的整个过程中,对scrapy框架运行的某些步骤做一些适配自己项目的动作. 例如scrapy内置的HttpErrorMiddleware,可以在http请求 ...

  8. 静态类和静态方法,抽象类和抽象方法,new关键字,值类型和引用类型,接口

    静态类和静态方法:静态成员是与类相关,而非实例相关:普通类中的静态成员:1.只能通过类名访问.2.静态方法中只能访问静态成员,或通过对象访问实例成员.3.多个对象共享同一个成员.静态类(一般用作工具类 ...

  9. Linux网络编程服务器模型选择之并发服务器(下)

    前面两篇文章(参见)分别介绍了循环服务器和简单的并发服务器网络模型,我们已经知道循环服务器模型效率较低,同一时刻只能为一个客户端提供服务,而且对于TCP模型来说,还存在单客户端长久独占与服务器的连接, ...

  10. dubbo接口快速测试技巧

    在分布式系统的开发中,用到了dubbo+zookeeper技术,最近遇到一个问题,产品上线后,我负责的模块出了问题,某个bean中某个字段的值一直为null,而这个bean是我调用注册在zookeep ...