k-d tree + 重构的思想,就能卡过luogu和bzoj啦orz

#include <algorithm>
#include <iostream>
#include <cstdio>
using namespace std;
int n, m, rot, nowD, opt, uu, vv, ans, rub[1000005], din, cur;
const double aph=0.65;
struct Point{
int d[2], mn[2], mx[2], l, r, siz;
int & operator[](int x){
return d[x];
}
bool operator<(const Point &x)const{
return d[nowD]<x.d[nowD];
}
Point(int x=0, int y=0){
l = r = siz = 0;
d[0] = x; d[1] = y;
}
}p[1000005];
int getDis(Point x, Point y){
return abs(x[1]-y[1])+abs(x[0]-y[0]);
}
struct KDTree{
Point t[1000005], T;
void pushUp(int x){
Point l=t[t[x].l], r=t[t[x].r];
for(int i=0; i<2; i++){
t[x].mn[i] = t[x].mx[i] = t[x][i];
if(t[x].l){
t[x].mn[i] = min(t[x].mn[i], l.mn[i]);
t[x].mx[i] = max(t[x].mx[i], l.mx[i]);
}
if(t[x].r){
t[x].mn[i] = min(t[x].mn[i], r.mn[i]);
t[x].mx[i] = max(t[x].mx[i], r.mx[i]);
}
}
t[x].siz = l.siz + r.siz + 1;
}
int newNode(){
if(din) return rub[din--];
else return ++cur;
}
int build(int l, int r, int now){
nowD = now;
int mid=(l+r)>>1, k=newNode();
nth_element(p+l, p+mid, p+1+r);
t[k] = p[mid];
t[k].l = t[k].r = t[k].siz = 0;
if(l<mid) t[k].l = build(l, mid-1, now^1);
if(mid<r) t[k].r = build(mid+1, r, now^1);
pushUp(k);
return k;
}
void pia(int k, int num){
if(t[k].l) pia(t[k].l, num);
p[num+t[t[k].l].siz+1] = t[k];
rub[++din] = k;
if(t[k].r) pia(t[k].r, num+t[t[k].l].siz+1);
}
void chkPia(int &k, int now){
if(aph*t[k].siz<t[t[k].l].siz || aph*t[k].siz<t[t[k].r].siz){
pia(k, 0);
k = build(1, t[k].siz, now);
}
}
void insert(int &k, int now){
if(!k){
k = newNode();
t[k] = T;
t[k].l = t[k].r = 0;
pushUp(k);
return ;
}
if(T[now]>=t[k][now]) insert(t[k].r, now^1);
else insert(t[k].l, now^1);
pushUp(k);
chkPia(k, now);
}
int get(int k, Point p){
int re=0;
for(int i=0; i<2; i++)
re += max(0, p[i]-t[k].mx[i]);
for(int i=0; i<2; i++)
re += max(0, t[k].mn[i]-p[i]);
return re;
}
void query(int k, int now){
int d=getDis(t[k],T), dl=0x3f3f3f3f, dr=0x3f3f3f3f;
ans = min(ans, d);
if(t[k].l) dl = get(t[k].l, T);
if(t[k].r) dr = get(t[k].r, T);
if(dl<dr){
if(dl<ans) query(t[k].l, now^1);
if(dr<ans) query(t[k].r, now^1);
}
else{
if(dr<ans) query(t[k].r, now^1);
if(dl<ans) query(t[k].l, now^1);
}
}
}kdt;
void rn(int &x){
char ch=getchar();
x = 0;
while(ch<'0' || ch>'9') ch = getchar();
while(ch>='0' && ch<='9'){
x = x * 10 + ch - '0';
ch = getchar();
}
}
int main(){
cin>>n>>m;
for(int i=1; i<=n; i++){
rn(p[i][0]);
rn(p[i][1]);
}
rot = kdt.build(1, n, 0);
while(m--){
scanf("%d %d %d", &opt, &uu, &vv);
kdt.T = Point(uu, vv);
if(opt==1)
kdt.insert(rot, 0);
else{
ans = 0x3f3f3f3f;
kdt.query(rot, 0);
printf("%d\n", ans);
}
}
return 0;
}

luogu4169 [Violet]天使玩偶/SJY摆棋子 / bzoj2648 SJY摆棋子 k-d tree的更多相关文章

  1. [Luogu4169][Violet]天使玩偶/SJY摆棋子

    luogu 题意 一个平面上有\(n\)个点,\(m\)次操作,每次新增一个点,或者是询问离某个点最近的点的距离.这里的距离是曼哈顿距离. \(n,m\le3*10^5\) sol 写一发\(CDQ\ ...

  2. bzoj2716/2648 / P4169 [Violet]天使玩偶/SJY摆棋子

    P4169 [Violet]天使玩偶/SJY摆棋子 k-d tree 模板 找了好几天才发现输出优化错了....真是zz...... 当子树非常不平衡时,就用替罪羊树的思想,拍扁重建. luogu有个 ...

  3. 【LG4169】[Violet]天使玩偶/SJY摆棋子

    [LG4169][Violet]天使玩偶/SJY摆棋子 题面 洛谷 题解 至于\(cdq\)分治的解法,以前写过 \(kdTree\)的解法好像还\(sb\)一些 就是记一下子树的横.纵坐标最值然后求 ...

  4. 洛谷 P4169 [Violet]天使玩偶/SJY摆棋子 解题报告

    P4169 [Violet]天使玩偶/SJY摆棋子 题目描述 \(Ayu\)在七年前曾经收到过一个天使玩偶,当时她把它当作时间囊埋在了地下.而七年后 的今天,\(Ayu\) 却忘了她把天使玩偶埋在了哪 ...

  5. luoguP4169 [Violet]天使玩偶/SJY摆棋子 K-Dtree

    P4169 [Violet]天使玩偶/SJY摆棋子 链接 luogu 思路 luogu以前用CDQ一直过不去. bzoj还是卡时过去的. 今天终于用k-dtree给过了. 代码 #include &l ...

  6. 洛谷P4169 [Violet]天使玩偶/SJY摆棋子(CDQ分治)

    [Violet]天使玩偶/SJY摆棋子 题目传送门 解题思路 用CDQ分治开了氧气跑过. 将输入给的顺序作为第一维的时间,x为第二维,y为第三维.对于距离一个询问(ax,ay),将询问分为四块,左上, ...

  7. [Violet]天使玩偶/SJY摆棋子 [cdq分治]

    P4169 [Violet]天使玩偶/SJY摆棋子 求离 \((x,y)\) 最近点的距离 距离的定义是 \(|x1-x2|+|y1-y2|\) 直接cdq 4次 考虑左上右上左下右下就可以了-略微卡 ...

  8. BZOJ2648/2716:SJY摆棋子/[Violet]天使玩偶(K-D Tree)

    Description 这天,SJY显得无聊.在家自己玩.在一个棋盘上,有N个黑色棋子.他每次要么放到棋盘上一个黑色棋子,要么放上一个白色棋子,如果是白色棋子,他会找出距离这个白色棋子最近的黑色棋子. ...

  9. P4169 [Violet]天使玩偶/SJY摆棋子

    题目背景 感谢@浮尘ii 提供的一组hack数据 题目描述 Ayu 在七年前曾经收到过一个天使玩偶,当时她把它当作时间囊埋在了地下.而七年后 的今天,Ayu 却忘了她把天使玩偶埋在了哪里,所以她决定仅 ...

随机推荐

  1. HTML5&CSS挑战

    地址:https://www.w3cschool.cn/codecamp/list?pename=html5_and_css_camp 开始学习HTML标签:欢迎来到编程训练营的第一个编程挑战!你可以 ...

  2. 开发环境中快速部署Oracle Essbase(Rapid deployment of oracle essbase in development envrioments)

    版本:Oracle Enterprise Performance Management System Release11.1.2.4 快速部署自动安装组件: Essbase Oracle Essbas ...

  3. python logging 模块记录日志

    #日志记录到多文件示例 import logging def error_log(message): file_1_1 = logging.FileHandler('error.log', 'a+', ...

  4. Coursera_Learn how to learn笔记

    番茄工作法,隔20分钟休息一次. 构建组块步骤:1.集中注意力. 2.理解基本概念. 3.进行练习. 获得专业知识的第一步是创建概念组块,能够将分散的信息集合到一起. Recall(回顾)比反复阅读更 ...

  5. [转载]正则表达式参考文档 - Regular Expression Syntax Reference.

    正则表达式参考文档 - Regular Expression Syntax Reference. [原创文章,转载请保留或注明出处:http://www.regexlab.com/zh/regref. ...

  6. constraint的一些用法总结

    主要就是增加约束的 以下几种约束 .并 一一列举: 1.主键约束: 要对一个列加主键约束的话,这列就必须要满足的条件就是分空 因为主键约束:就是对一个列进行了约束,约束为(非空.不重复) 以下是代码  ...

  7. IOS NSKeyedArchiver(归档存取数据)

    如果对象是NSString.NSDictionary.NSArray.NSData.NSNumber等类 型,可以直接用NSKeyedArchiver进行归档和恢复 不是所有的对象都可以直接用这种方法 ...

  8. 【转】Nginx搭建反向代理服务器过程详解

    阅读目录 1.1 反向代理初印象 1.2 反向代理的作用 2.1 Nginx是神马? 2.2 Nginx的应用现状 2.3 Nginx的核心特点 3.1 准备一个ASP.NET网站部署到IIS服务器集 ...

  9. iterable -------JavaScript

    本文摘要:http://www.liaoxuefeng.com/ 遍历Array可以采用下标循环,遍历Map和Set就无法使用下标.为了统一集合类型,ES6标准引入了新的iterable类型,Arra ...

  10. Bootstrap 折叠(collapse)插件面板

    折叠插件(collapse)可以很容易地让页面区域折叠起来, 无论您是用它来创建折叠导航还是内容面板,它都允许很多内容选项. 您可以使用折叠插件 1.创建可折叠的分组或折叠的面板 <!DOCTY ...