欢迎访问 生活随笔!

尊龙凯时首页

当前位置: 尊龙凯时首页 > 编程资源 > 编程问答 >内容正文

编程问答

kd tree学习笔记 (最近邻域查询) -尊龙凯时首页

发布时间:2023/12/18 编程问答 14 豆豆
尊龙凯时首页 收集整理的这篇文章主要介绍了 kd tree学习笔记 (最近邻域查询) 小编觉得挺不错的,现在分享给大家,帮大家做个参考.

https://zhuanlan.zhihu.com/p/22557068

http://blog.csdn.net/zhjchengfeng5/article/details/7855241

 

kd树在算法竞赛中主要用来做各种各样的平面区域查询,包含则累加直接返回,相交则继续递归,相离的没有任何贡献也直接返回。可以处理圆,三角形,矩形等判断起来相对容易的平面区域内的符合加法性质的操作。

比如查询平面内欧几里得距离最近的点的距离。

kdtree其实有点像搜索,暴力 剪枝。

每次从根结点向下搜索,并进行剪枝操作,判断是否有必要继续搜索。

它是通过横一刀,竖一刀,横一刀再竖一刀将平面进行分割,建立二叉树。

建树的复杂度是o(nlogn), 每次用nth_element()在线性时间内取出中位数。 t(n) = 2t(n/2) o(n) = o(nlogn)

查询复杂度呢? 据第二个链接的博客说最坏是o( sqrt(n) ) 的。并不会分析查询复杂度。

 

hdu2966 裸kdtree

题意:给平面图上n(≤ 100000)个点,对每个点,找到其他 欧几里德距离 离他最近的点,输出他们之间的距离。保证没有重点。

1 #include 2 #define ll long long 3 using namespace std; 4 #define n 200010 5 const ll inf = 1e18; 6 int n,i,id[n],root,cmp_d; 7 int x, y; 8 struct node{int d[2],l,r,max[2],min[2],val,sum,f;}t[n]; 9 bool cmp(const node&a,const node&b){return a.d[cmp_d]<b.d[cmp_d];} 10 void umax(int&a,int b){if(ab;} 11 void umin(int&a,int b){if(a>b)a=b;} 12 void up(int x){ 13 if(t[x].l){ 14 umax(t[x].max[0],t[t[x].l].max[0]); 15 umin(t[x].min[0],t[t[x].l].min[0]); 16 umax(t[x].max[1],t[t[x].l].max[1]); 17 umin(t[x].min[1],t[t[x].l].min[1]); 18 } 19 if(t[x].r){ 20 umax(t[x].max[0],t[t[x].r].max[0]); 21 umin(t[x].min[0],t[t[x].r].min[0]); 22 umax(t[x].max[1],t[t[x].r].max[1]); 23 umin(t[x].min[1],t[t[x].r].min[1]); 24 } 25 } 26 int build(int l,int r,int d,int f){ 27 int mid=(l r)>>1; 28 cmp_d=d,std::nth_element(t l,t mid,t r 1,cmp); 29 id[t[mid].f]=mid; 30 t[mid].f=f; 31 t[mid].max[0]=t[mid].min[0]=t[mid].d[0]; 32 t[mid].max[1]=t[mid].min[1]=t[mid].d[1]; 33 //t[mid].val=t[mid].sum=0; 34 if(l!=mid)t[mid].l=build(l,mid-1,!d,mid);else t[mid].l=0; 35 if(r!=mid)t[mid].r=build(mid 1,r,!d,mid);else t[mid].r=0; 36 return up(mid),mid; 37 } 38 39 ll dis(ll x1, ll y1, ll x, ll y) { 40 ll xx = x1-x, yy = y1-y; 41 return xx*xx yy*yy; 42 } 43 ll dis(int p, ll x, ll y){//估价函数, 以p为子树的最小距离 44 ll xx = 0, yy = 0; 45 if(t[p].max[0] < x) xx = x-t[p].max[0]; 46 if(t[p].min[0] > x) xx = t[p].min[0]-x; 47 if(t[p].max[1] < y) yy = y-t[p].max[1]; 48 if(t[p].min[1] > y) yy = t[p].min[1]-y; 49 return xx*xx yy*yy; 50 } 51 ll ans; 52 void query(int p){ 53 ll dl = inf, dr = inf, d = dis(t[p].d[0], t[p].d[1], x, y); 54 if(d) ans = min(ans, d); 55 56 if(t[p].l) dl = dis(t[p].l, x, y); 57 if(t[p].r) dr = dis(t[p].r, x, y); 58 if(dl < dr){ 59 if(dl < ans) query(t[p].l); 60 if(dr < ans) query(t[p].r); 61 } 62 else { 63 if(dr < ans) query(t[p].r); 64 if(dl < ans) query(t[p].l); 65 } 66 } 67 68 int main(){ 69 int t; scanf("%d", &t); 70 while(t--){ 71 scanf("%d", &n); 72 for(int i = 1; i <= n; i ){ 73 scanf("%d%d", &t[i].d[0], &t[i].d[1]); 74 t[i].f = i; 75 } 76 int rt = build(1, n, 0, 0); 77 for(int i = 1; i <= n; i ){ 78 ans = inf; 79 x = t[ id[i] ].d[0], y = t[ id[i] ].d[1]; 80 query(rt); 81 printf("%lld\n", ans); 82 } 83 } 84 return 0; 85 } view code

 

bzoj2648

题意:给出n个点,接下来m个操作,每次插入一个点,或者询问离询问点的最近曼哈顿距离。

1 #include 2 #define ll long long 3 using namespace std; 4 #define n 1000010 5 const ll inf = 1e18; 6 int n,m,i,id[n],root,cmp_d,rt; 7 int x, y; 8 struct node{int d[2],l,r,max[2],min[2],val,sum,f;}t[n]; 9 bool cmp(const node&a,const node&b){return a.d[cmp_d]<b.d[cmp_d];} 10 void umax(int&a,int b){if(ab;} 11 void umin(int&a,int b){if(a>b)a=b;} 12 void up(int x){ 13 if(t[x].l){ 14 umax(t[x].max[0],t[t[x].l].max[0]); 15 umin(t[x].min[0],t[t[x].l].min[0]); 16 umax(t[x].max[1],t[t[x].l].max[1]); 17 umin(t[x].min[1],t[t[x].l].min[1]); 18 } 19 if(t[x].r){ 20 umax(t[x].max[0],t[t[x].r].max[0]); 21 umin(t[x].min[0],t[t[x].r].min[0]); 22 umax(t[x].max[1],t[t[x].r].max[1]); 23 umin(t[x].min[1],t[t[x].r].min[1]); 24 } 25 } 26 int build(int l,int r,int d,int f){ 27 int mid=(l r)>>1; 28 cmp_d=d,std::nth_element(t l,t mid,t r 1,cmp); 29 id[t[mid].f]=mid; 30 t[mid].f=f; 31 t[mid].max[0]=t[mid].min[0]=t[mid].d[0]; 32 t[mid].max[1]=t[mid].min[1]=t[mid].d[1]; 33 //t[mid].val=t[mid].sum=0; 34 if(l!=mid)t[mid].l=build(l,mid-1,!d,mid);else t[mid].l=0; 35 if(r!=mid)t[mid].r=build(mid 1,r,!d,mid);else t[mid].r=0; 36 return up(mid),mid; 37 } 38 39 ll dis(ll x1, ll y1, ll x, ll y) { 40 return abs(x1-x) abs(y1-y); 41 //ll xx = x1-x, yy = y1-y; 42 //return xx*xx yy*yy; 43 } 44 ll dis(int p, ll x, ll y){//估价函数, 以p为子树的最小距离 45 ll xx = 0, yy = 0; 46 if(t[p].max[0] < x) xx = x-t[p].max[0]; 47 if(t[p].min[0] > x) xx = t[p].min[0]-x; 48 if(t[p].max[1] < y) yy = y-t[p].max[1]; 49 if(t[p].min[1] > y) yy = t[p].min[1]-y; 50 return xx yy; 51 //return xx*xx yy*yy; 52 } 53 ll ans; 54 void ins(int now, int k, int x){ 55 if(t[x].d[k] >= t[now].d[k]){ 56 if(t[now].r) ins(t[now].r, !k, x); 57 else 58 t[now].r = x, t[x].f = now; 59 } 60 else { 61 if(t[now].l) ins(t[now].l, !k, x); 62 else t[now].l = x, t[x].f = now; 63 } 64 up(now); 65 } 66 void query(int p){ 67 ll dl = inf, dr = inf, d = dis(t[p].d[0], t[p].d[1], x, y); 68 ans = min(ans, d); 69 70 if(t[p].l) dl = dis(t[p].l, x, y); 71 if(t[p].r) dr = dis(t[p].r, x, y); 72 if(dl < dr){ 73 if(dl < ans) query(t[p].l); 74 if(dr < ans) query(t[p].r); 75 } 76 else { 77 if(dr < ans) query(t[p].r); 78 if(dl < ans) query(t[p].l); 79 } 80 } 81 82 int main(){ 83 scanf("%d%d", &n, &m); 84 for(int i = 1; i <= n; i ) 85 scanf("%d%d", &t[i].d[0], &t[i].d[1]); 86 rt = build(1, n, 0, 0); 87 while(m--){ 88 int op; 89 scanf("%d%d%d", &op, &x, &y); 90 if(op == 1){ 91 n ; 92 t[n].l = t[n].r = 0; 93 t[n].max[0] = t[n].min[0] = t[n].d[0] = x; 94 t[n].max[1] = t[n].min[1] = t[n].d[1] = y; 95 ins(rt, 0, n); 96 } 97 else{ 98 ans = inf; 99 query(rt); 100 printf("%lld\n", ans); 101 } 102 } 103 return 0; 104 } view code

 

bzoj3053

题意:k维坐标系下的最近的m个点。直接对于每一个询问都在kdtree中询问m次最近点,每次找到一个最近点对需要把它记录下来,用堆维护即可。

1 #include 2 #define ll long long 3 #define mp make_pair 4 5 using namespace std; 6 #define n 50010 7 const ll inf = 1e18; 8 int n,m,k,i,id[n],root,cmp_d,rt; 9 int x, y, num; 10 struct node{int d[5],l,r,max[5],min[5],val,sum,f;}t[n]; 11 bool cmp(const node&a,const node&b){return a.d[cmp_d]<b.d[cmp_d];} 12 void umax(int&a,int b){if(ab;} 13 void umin(int&a,int b){if(a>b)a=b;} 14 void up(int x){ 15 for(int i = 0; i < k; i ){ 16 if(t[x].l){ 17 umax(t[x].max[i],t[t[x].l].max[i]); 18 umin(t[x].min[i],t[t[x].l].min[i]); 19 } 20 if(t[x].r){ 21 umax(t[x].max[i],t[t[x].r].max[i]); 22 umin(t[x].min[i],t[t[x].r].min[i]); 23 } 24 } 25 } 26 int build(int l,int r,int d,int f){ 27 int mid=(l r)>>1; 28 cmp_d=d,std::nth_element(t l,t mid,t r 1,cmp); 29 id[t[mid].f]=mid; 30 t[mid].f=f; 31 for(int i = 0; i < k; i ) 32 t[mid].max[i]=t[mid].min[i]=t[mid].d[i]; 33 //t[mid].max[1]=t[mid].min[1]=t[mid].d[1]; 34 //t[mid].val=t[mid].sum=0; 35 if(l!=mid)t[mid].l=build(l,mid-1,(d 1)%k,mid);else t[mid].l=0; 36 if(r!=mid)t[mid].r=build(mid 1,r,(d 1)%k,mid);else t[mid].r=0; 37 return up(mid),mid; 38 } 39 int qx[5]; 40 ll dis(int p){//估价函数, 以p为子树的最小距离 41 ll ret = 0, ans = 0; 42 for(int i = 0; i < k; i ) { 43 ret = 0; 44 if(t[p].max[i] < qx[i]) ret = qx[i]-t[p].max[i]; 45 if(t[p].min[i] > qx[i]) ret = t[p].min[i]-qx[i]; 46 ans = ret*ret; 47 } 48 return ans; 49 } 50 ll getdis(int p){ 51 ll ans = 0; 52 for(int i = 0; i < k; i ) 53 ans = (qx[i]-t[p].d[i])*(qx[i]-t[p].d[i]); 54 return ans; 55 } 56 void ins(int now, int k, int x){ 57 if(t[x].d[k] >= t[now].d[k]){ 58 if(t[now].r) ins(t[now].r, !k, x); 59 else 60 t[now].r = x, t[x].f = now; 61 } 62 else { 63 if(t[now].l) ins(t[now].l, !k, x); 64 else t[now].l = x, t[x].f = now; 65 } 66 up(now); 67 } 68 ll ret; 69 multiset< pair<int, int> > ans; 70 void query(int p){ 71 ll dl = inf, dr = inf, d = getdis(p); 72 ans.insert( mp((int)d, p) ); 73 if(ans.size() > num){ 74 multiset< pair<int, int> >::iterator it = ans.end(); 75 it--; 76 ans.erase(it); 77 } 78 ret = (*ans.rbegin()).first; 79 if(t[p].l) dl = dis(t[p].l); 80 if(t[p].r) dr = dis(t[p].r); 81 if(dl < dr){ 82 if(dl < ret||ans.size() < num) query(t[p].l); 83 if(dr < ret||ans.size() < num) query(t[p].r); 84 } 85 else { 86 if(dr < ret||ans.size() < num) query(t[p].r); 87 if(dl < ret||ans.size() < num) query(t[p].l); 88 } 89 } 90 91 int main(){ 92 while(~scanf("%d%d", &n, &k)){ 93 for(int i = 1; i <= n; i ){ 94 for(int j = 0; j < k; j ) 95 scanf("%d", &t[i].d[j]); 96 } 97 rt = build(1, n, 0, 0); 98 scanf("%d", &m); 99 while(m--){ 100 for(int i = 0; i < k; i ) 101 scanf("%d", qx i); 102 scanf("%d", &num); 103 ans.clear(); 104 query(rt); 105 printf ("the closest %d points are:\n", num); 106 for(multiset< pair<int, int> >::iterator it = ans.begin(); it != ans.end(); it ){ 107 int pos = (*it).second; 108 for(int i = 0; i < k; i ) 109 printf("%d%c", t[pos].d[i], " \n"[i == k-1]); 110 } 111 } 112 } 113 return 0; 114 } view code

 

转载于:https://www.cnblogs.com/dirge/p/6091241.html

总结

以上是尊龙凯时首页为你收集整理的kd tree学习笔记 (最近邻域查询)的全部内容,希望文章能够帮你解决所遇到的问题。

如果觉得尊龙凯时首页网站内容还不错,欢迎将尊龙凯时首页推荐给好友。

  • 上一篇:
  • 下一篇:
网站地图