用于动态插入以及求某点的最近点的距离(BZOJ2648,BZOJ2716)

#include <cstdio>
#include <cmath>
#include <algorithm>
using namespace std; int cnt,ans,n,m; struct data{
int x,y;
}point[]; struct kd_node{
int x,y,lc,rc,minx,miny,maxx,maxy;
}tr[]; int dis(int x1,int y1,int x2,int y2){
return(fabs(x1-x2)+fabs(y1-y2));
} int mycomp1(const data&a,const data&b){
return(a.y<b.y);
} int mycomp2(const data&a,const data&b){
return(a.x<b.x);
} void kd_update(int po){
if (tr[po].lc){
tr[po].minx=min(tr[po].minx,tr[tr[po].lc].minx);
tr[po].miny=min(tr[po].miny,tr[tr[po].lc].miny);
tr[po].maxx=max(tr[po].maxx,tr[tr[po].lc].maxx);
tr[po].maxy=max(tr[po].maxy,tr[tr[po].lc].maxy);
};
if (tr[po].rc){
tr[po].minx=min(tr[po].minx,tr[tr[po].rc].minx);
tr[po].miny=min(tr[po].miny,tr[tr[po].rc].miny);
tr[po].maxx=max(tr[po].maxx,tr[tr[po].rc].maxx);
tr[po].maxy=max(tr[po].maxy,tr[tr[po].rc].maxy);
}
}//维护包含子树中所有点的最小的矩形,左下角(minx,miny),右上角(maxx,maxy) void kd_build(int l,int r,int wd){
if (wd) sort(point+l,point+r+,mycomp1);else sort(point+l,point+r+,mycomp2);//实际应选取方差最大的一维
cnt++;
int mid=(l+r)>>,t=cnt;//mid实际应为与平均值最接近的
tr[cnt].x=point[mid].x;tr[cnt].y=point[mid].y;
tr[cnt].minx=point[mid].x;tr[cnt].maxx=point[mid].x;
tr[cnt].miny=point[mid].y;tr[cnt].maxy=point[mid].y;
if (l<mid){
tr[t].lc=cnt+;kd_build(l,mid-,!wd);
};
if (mid<r){
tr[t].rc=cnt+;kd_build(mid+,r,!wd);
}
kd_update(t);
}//构建n点的KD树,每次以一维为标准划分点 void kd_ins(int po,int x,int y,int d){
int son=;
if (d==) son=(x<=tr[po].x);else son=(y<=tr[po].y);
if (son==){
if (tr[po].lc==){
tr[po].lc=++cnt;
tr[cnt].x=x;tr[cnt].y=y;
tr[cnt].minx=x;tr[cnt].maxx=x;
tr[cnt].miny=y;tr[cnt].maxy=y;
}else kd_ins(tr[po].lc,x,y,!d);
}else{
if (tr[po].rc==){
tr[po].rc=++cnt;
tr[cnt].x=x;tr[cnt].y=y;
tr[cnt].minx=x;tr[cnt].maxx=x;
tr[cnt].miny=y;tr[cnt].maxy=y;
}else kd_ins(tr[po].rc,x,y,!d);
}
kd_update(po);
} int dist(int x,int y,int po){
int ret=;
ret+=max(,tr[po].minx-x);
ret+=max(,x-tr[po].maxx);
ret+=max(,tr[po].miny-y);
ret+=max(,y-tr[po].maxy);
return(ret);
}//(x,y)到po子树维护的矩形的最近距离。假设该矩形中铺满点,所以返回值小于等于实际值 void kd_query(int po,int x,int y){
ans=min(ans,dis(x,y,tr[po].x,tr[po].y));
int dl=(tr[po].lc== ? 1e9:dist(x,y,tr[po].lc));
int dr=(tr[po].rc== ? 1e9:dist(x,y,tr[po].rc));
if (dl<dr){
if (dl<ans) kd_query(tr[po].lc,x,y);
if (dr<ans) kd_query(tr[po].rc,x,y);
}else{
if (dr<ans) kd_query(tr[po].rc,x,y);
if (dl<ans) kd_query(tr[po].lc,x,y);
}
}//以dist为估价函数搜寻 int main(){
scanf("%d%d",&n,&m);
for (int i=;i<=n;i++) scanf("%d%d",&point[i].x,&point[i].y);
kd_build(,n,); for (int i=;i<=m;i++){
int opt,x,y;
scanf("%d%d%d",&opt,&x,&y); if (opt==){
kd_ins(,x,y,);
}else{
ans=1e9;
kd_query(,x,y);
printf("%d\n",ans);
}
}
}

____________________________________________

BZOJ4066

单点修改,矩形求和

#include <cstdio>
#include <algorithm>
#include <iostream>
using namespace std; int cnt,tcnt,n; struct data{
int x,y,v;
}a[]; struct treenode{
int lc,rc,x,y,minx,miny,maxx,maxy,num,v;
}tr[]; void update(int po){
tr[po].num=tr[tr[po].lc].num+tr[tr[po].rc].num+tr[po].v;
tr[po].minx=min(min(tr[tr[po].lc].minx,tr[tr[po].rc].minx),tr[po].x);
tr[po].miny=min(min(tr[tr[po].lc].miny,tr[tr[po].rc].miny),tr[po].y);
tr[po].maxx=max(max(tr[tr[po].lc].maxx,tr[tr[po].rc].maxx),tr[po].x);
tr[po].maxy=max(max(tr[tr[po].lc].maxy,tr[tr[po].rc].maxy),tr[po].y);
} void insert(int po,int x,int y,int num,int wd){
if (tr[po].x==x&&tr[po].y==y){
tr[po].num+=num;tr[po].v+=num;return;
} int sel;
if (!wd) sel=(tr[po].x<x);else sel=(tr[po].y<y);
if (!sel){
if (!tr[po].lc){
cnt++;
tr[cnt].x=tr[cnt].minx=tr[cnt].maxx=x;
tr[cnt].y=tr[cnt].miny=tr[cnt].maxy=y;
tr[cnt].v=tr[cnt].num=num;
tr[po].lc=cnt;
update(po);
}else insert(tr[po].lc,x,y,num,!wd);
}else{
if (!tr[po].rc){
cnt++;
tr[cnt].x=tr[cnt].minx=tr[cnt].maxx=x;
tr[cnt].y=tr[cnt].miny=tr[cnt].maxy=y;
tr[cnt].v=tr[cnt].num=num;
tr[po].rc=cnt;
update(po);
}else insert(tr[po].rc,x,y,num,!wd);
}
update(po);
} int mycomp1(const data&a,const data&b){
return(a.x<b.x);
} int mycomp2(const data&a,const data&b){
return(a.y<b.y);
} void build(int l,int r,int wd){
if (wd==) sort(a+l,a+r+,mycomp1);else sort(a+l,a+r+,mycomp2);
int mid=(l+r)>>;
int tmp=++tcnt;
tr[tcnt].lc=tr[tcnt].rc=;
tr[tcnt].x=a[mid].x;tr[tcnt].y=a[mid].y;tr[tcnt].v=a[mid].v;
if (l<mid){
tr[tmp].lc=tcnt+;build(l,mid-,!wd);
};
if (r>mid){
tr[tmp].rc=tcnt+;build(mid+,r,!wd);
}
update(tmp);
} int in(int x1,int y1,int x2,int y2,int X1,int Y1,int X2,int Y2){
return(X1<=x1&&Y1<=y1&&X2>=x2&&Y2>=y2);
} int out(int x1,int y1,int x2,int y2,int X1,int Y1,int X2,int Y2){
return(x2<X1||x1>X2||y1>Y2||y2<Y1);
} int query(int po,int x1,int y1,int x2,int y2){
int ret=;
if (in(tr[po].minx,tr[po].miny,tr[po].maxx,tr[po].maxy,x1,y1,x2,y2)) return(tr[po].num); if (in(tr[po].x,tr[po].y,tr[po].x,tr[po].y,x1,y1,x2,y2)) ret+=tr[po].v;
if (!out(tr[tr[po].lc].minx,tr[tr[po].lc].miny,tr[tr[po].lc].maxx,tr[tr[po].lc].maxy,x1,y1,x2,y2)) ret+=query(tr[po].lc,x1,y1,x2,y2);
if (!out(tr[tr[po].rc].minx,tr[tr[po].rc].miny,tr[tr[po].rc].maxx,tr[tr[po].rc].maxy,x1,y1,x2,y2)) ret+=query(tr[po].rc,x1,y1,x2,y2);
return(ret);
} int main(){
scanf("%d",&n);
tr[].miny=tr[].minx=1e9;
tr[].maxx=tr[].maxy=-1e9;
int opt,lastans=,lastrebuild=,root=;
while (scanf("%d",&opt),opt!=){
int x,y,x1,y1,x2,y2,num; if (opt==){
scanf("%d%d%d",&x,&y,&num);
x^=lastans;y^=lastans;num^=lastans;
if (!root){
root=;
cnt++;
tr[cnt].x=tr[cnt].minx=tr[cnt].maxx=x;
tr[cnt].y=tr[cnt].miny=tr[cnt].maxy=y;
tr[cnt].v=tr[cnt].num=num;
}else{
insert(root,x,y,num,);
} if (cnt/>lastrebuild){
for (int i=;i<=cnt;i++) a[i].x=tr[i].x,a[i].y=tr[i].y,a[i].v=tr[i].v;
tcnt=;
build(,cnt,);
lastrebuild++;
}
} if (opt==){
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
x1^=lastans;x2^=lastans;y1^=lastans;y2^=lastans;
lastans=query(,x1,y1,x2,y2);
printf("%d\n",lastans);
}
}
}

KD-tree(2维)的更多相关文章

  1. Wannafly Winter Camp 2020 Day 5I Practice for KD Tree - 二维线段树

    给定一个 \(n \times n\) 矩阵,先进行 \(m_1 \leq 5e4\) 次区间加,再进行 \(m_2 \leq 5e5\) 次询问,每次询问要求输出矩形区间内的最大数.\(n \leq ...

  2. k-d tree 学习笔记

    以下是一些奇怪的链接有兴趣的可以看看: https://blog.sengxian.com/algorithms/k-dimensional-tree http://zgjkt.blog.uoj.ac ...

  3. k-d tree算法

    k-d树(k-dimensional树的简称),是一种分割k维数据空间的数据结构.主要应用于多维空间关键数据的搜索(如:范围搜索和最近邻搜索). 应用背景 SIFT算法中做特征点匹配的时候就会利用到k ...

  4. [模板] K-D Tree

    K-D Tree K-D Tree可以看作二叉搜索树的高维推广, 它的第 \(k\) 层以所有点的第 \(k\) 维作为关键字对点做出划分. 为了保证划分均匀, 可以以第 \(k\) 维排名在中间的节 ...

  5. HDU2966 In case of failure(浅谈k-d tree)

    嘟嘟嘟 题意:给定\(n\)个二维平面上的点\((x_i, y_i)\),求离每一个点最近的点得距离的平方.(\(n \leqslant 1e5\)) 这就是k-d tree入门题了. k-d tre ...

  6. [学习笔记]K-D Tree

    以前其实学过的但是不会拍扁重构--所以这几天学了一下 \(K-D\ Tree\) 的正确打开姿势. \(K\) 维 \(K-D\ Tree\) 的单次操作最坏时间复杂度为 \(O(k\times n^ ...

  7. BZOJ 3053: The Closest M Points(K-D Tree)

    Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1235  Solved: 418[Submit][Status][Discuss] Descripti ...

  8. 浅谈K-D Tree

    初步认识\(K-D\) \(Tree\) \(K-D\) \(Tree\)是一种基于空间分割的二叉树形数据结构,一般用于高维信息检索.因为\(OI\)中很多问题都能转化为高维信息检索,所以\(K-D\ ...

  9. K-D TREE算法原理及实现

    博客转载自:https://leileiluoluo.com/posts/kdtree-algorithm-and-implementation.html k-d tree即k-dimensional ...

  10. [转载]kd tree

    [本文转自]http://www.cnblogs.com/eyeszjwang/articles/2429382.html k-d树(k-dimensional树的简称),是一种分割k维数据空间的数据 ...

随机推荐

  1. 【DDD/CQRS/微服务架构案例】在Ubuntu 14.04.4 LTS中运行WeText项目的服务端

    在<WeText项目:一个基于.NET实现的DDD.CQRS与微服务架构的演示案例>文章中,我介绍了自己用Visual Studio 2015(C# 6.0 with .NET Frame ...

  2. LinqToDB 源码分析——DataContext类

    LinqToDB框架是一个轻量级的ORM框架.当然,功能上来讲一定比不上Entity Framework的强大.但是在使用上总让笔者感觉有一点Entity Framework的影子.笔者想过可能的原因 ...

  3. linux sudo 命令

    简单的说,sudo 是一种权限管理机制,管理员可以授权于一些普通用户去执行一些 root 执行的操作,而不需要知道 root 的密码.严谨些说,sudo 允许一个已授权用户以超级用户或者其它用户的角色 ...

  4. <input type="file">上传文件并添加路径到数据库

    注:这里是用的mvc所以没法用控件 html代码 <form method="post" enctype="multipart/form-data"> ...

  5. 手把手教从零开始在GitHub上使用Hexo搭建博客教程(一)-附GitHub注册及配置

    前言 有朋友问了我关于博客系统搭建相关的问题,由于是做开发相关的工作,我给他推荐的是使用github的gh-pages服务搭建个人博客. 推荐理由: 免费:github提供gh-pages服务是免费的 ...

  6. WebApi安全性 使用TOKEN+签名验证

    首先问大家一个问题,你在写开放的API接口时是如何保证数据的安全性的?先来看看有哪些安全性问题在开放的api接口中,我们通过http Post或者Get方式请求服务器的时候,会面临着许多的安全性问题, ...

  7. 记录一次bug解决过程:可维护性和性能优化

    一.总结 使用某些变量的地方在2次以上的,强烈建议使用枚举值来维护变量,日后方便扩展. 查数据库的方法调用,能合并就净量去合并. 二.Bug描述 枚举变量的维护以及方法使用: public class ...

  8. php实现设计模式之 状态模式

    <?php /*状态模式:允许一个对象在其内部状态改变时改变它的行为.对象看起来似乎修改了它的类.(行为模式) * * 在很多情况下,一个对象的行为取决于一个或多个动态变化的属性,这样的属性叫做 ...

  9. K60——寄存器

    (1)PTx_BASE_PTR为GPIO寄存器结构体基址指针(PTR即point to register,x=A/B/C/D/E) /* GPIO - Peripheral instance base ...

  10. JavaScript和jQuery的类型判断

    此博文为原创,转载请注明出处! 对于类型的判断,JavaScript用typeof来进行. 栗子: console.log(typeof null); //object console.log(typ ...