zoukankan      html  css  js  c++  java
  • K-D Tree

    K-D Tree可以搞多维空间问题,其形式为一棵二叉搜索树,它能把一张高维(>=2)的图分成好多块,其节点的某维坐标大于其左儿子,小于其右儿子。

    K-D Tree的建立

    对于k维的问题,第i层我们根据区间内各点的第(i % k)维坐标,用快速排序的思想,可以做到 log 时间内找中位数,然后以中位数的节点为当前根,把当前区间一分为二,然后(如果有的话)递归到其左右区间,左右区间的根即为当前根的左右儿子。如图:

    k-d tree形态

    【卡常技巧】最好把 (ls,rs,val,siz) 之类的一块放在结构体里面,据说会因为“连续”而变快。

    (Code:)

    //二维k-d tree
    struct kdtree{
    	int mn[2], mx[2], d[2];
    	int ls, rs, val;
    	kdtree() {
    		ls = rs = val = d[0] = d[1] = 0;
    		mn[0] = mn[1] = inf;
    		mx[0] = mx[1] = 0;
    	}
    }kdt[N];
    bool cmp(const kdtree &a, const kdtree &b) {
    	return a.d[type] < b.d[type];
    }
    void build(int L, int R, int k, int &cur) {
    	int mid = (L + R) >> 1;
    	cur = mid;
    	type = k;
    	nth_element(kdt + L + 1, kdt + mid + 1, kdt + R + 1, cmp);
    	if (mid - 1 >= L)	build(L, mid - 1, k ^ 1, kdt[mid].ls);
    	if (mid + 1 <= R) build(mid + 1, R, k ^ 1, kdt[mid].rs);
    	pushup(mid);//依据题意写
    }
    

    K-D Tree的复杂度

    K-D Tree实际上是一种暴力的优化算法,它的使用就是暴力加剪枝,但复杂度竟然是n^(1+(1 - 1/k))(k维)

    K-D Tree的使用(例题)

    P4475 巧克力王国

    以x和y为横纵坐标,建立平面直角坐标系。注意到对于每个询问,其符合要求的范围是连续的。确切地说,其范围应该是一条直线的左端。建一颗K-D Tree,然后从根节点开始找,如果某节点全部符合要求或全部不符合要求,就直接把它剪掉,不往下递归。这需要我们维护各节点的美味度总和.

    部分代码:

    inline void pushup(int cur) {
    	register int ls = kdt[cur].ls, rs = kdt[cur].rs;
    	kdt[cur].sum = kdt[ls].sum + kdt[rs].sum + kdt[cur].val;
    	for (register int i = 0; i <= 1; ++i) {
    		kdt[cur].mx[i] = kdt[cur].mn[i] = kdt[cur].d[i];
    		if (ls) {
    			kdt[cur].mn[i] = min(kdt[ls].mn[i], kdt[cur].mn[i]);
    			kdt[cur].mx[i] = max(kdt[ls].mx[i], kdt[cur].mx[i]);
    		}
    		if (rs) {
    			kdt[cur].mn[i] = min(kdt[rs].mn[i], kdt[cur].mn[i]);
    			kdt[cur].mx[i] = max(kdt[rs].mx[i], kdt[cur].mx[i]);
    		}
    	}
    }
    ...
    inline bool che(int x, int y) {
    	return aaa * x + bbb * y < ccc;
    }
    int query(int cur) {
    	int res = 0, cnt = 0;
    	cnt += che(kdt[cur].mn[0], kdt[cur].mn[1]);
    	cnt += che(kdt[cur].mn[0], kdt[cur].mx[1]);
    	cnt += che(kdt[cur].mx[0], kdt[cur].mn[1]);
    	cnt += che(kdt[cur].mx[0], kdt[cur].mx[1]);
    	if (cnt == 4)	return kdt[cur].sum;
    	if (!cnt)	return 0;
    	if (che(kdt[cur].d[0], kdt[cur].d[1]))	res += kdt[cur].val;
    	if (kdt[cur].ls)	res += query(kdt[cur].ls);
    	if (kdt[cur].rs)	res += query(kdt[cur].rs);
    	return res;
    }
    

    P4357 [CQOI2016]K远点对

    搞个小根堆,维护最大的那k个点对。

    由于K-D Tree 的剪枝像大多数 DFS 的剪枝一样,它并不需要一些准确的信息,只要“最优”情况不能更新答案,就可以剪掉它。不过更新答案的时候是要用准确信息的。

    这道题的“最优”情况为矩形的四条边(甚至都可能不是一个点)的坐标。

    加强版:P2093 [国家集训队]JZPFAR

    我做的第一道国集JZP题

    其实这种问题还能优化。查询的时候,如果发现左儿子的最优假答案比右儿子的最优假答案更优的话,那么我们要先去左儿子,再去右儿子。这样,我们先获得了更接近最优答案的答案,以后就能剪掉更多的枝了。

    这个剪枝是个 K-D Tree 的经典套路。这道题(JZPFAR)不这么剪还有一半分,P2479 [SDOI2010]捉迷藏就只有30分了。

    关键代码:

    //pr = pair,一开始想用pair水过,后来还是写的结构体
    inline ll get_dis(int x, int y, int X, int Y) {
    	return Pow(x - X) + Pow(y - Y);
    }
    inline ll fake_dis(node nd, int x, int y) {
    	return max(Pow(nd.mx[0] - x), Pow(nd.mn[0] - x)) + 
    		max(Pow(nd.mx[1] - y), Pow(nd.mn[1] - y));
    }
    void query(int x, int y, int cur) {
    	if (!cur)	return ;
    	ll tmp = get_dis(nd[cur].d[0], nd[cur].d[1], x, y);
    	Node pr = (Node){tmp, nd[cur].id};
    	if (pr < q.top())	q.pop(), q.push(pr);
    	int ls = nd[cur].ls, rs = nd[cur].rs;
    	ll dl, dr;
    	if (ls) dl = fake_dis(nd[ls], x, y);
    	else	dl = -1;
    	if (rs)	dr = fake_dis(nd[rs], x, y);
    	else	dr = -1;
    	Node Pl = (Node){dl, nd[ls].id}, Pr = (Node){dr, nd[rs].id};
    	if (Pl < Pr) {
    		if (Pl < q.top())	query(x, y, ls);
    		if (Pr < q.top())	query(x, y, rs);
    	} else {
    		if (Pr < q.top())	query(x, y, rs);
    		if (Pl < q.top())	query(x, y, ls);
    	}
    }
    

    P4148 简单题

    二维平面中单点加,矩形数点。强制在线。(q <= 2e5).空间20MB,时间8s.

    由于强制在线且卡空间,这题 K-D Tree 成为比较理想的解法。

    由于不断加点可能导致不平衡,我们需要不时地重构一下。或者像替罪羊树那样搞一个 (alpha) 值。

    void Build(int L, int R, int d, int &cur) {
    	if (L > R)	return cur = 0, void();
    	int mid = (L + R) >> 1;
    	nwd = d;
    	nth_element(stk + L, stk + mid, stk + R, cmp);
    	nd[cur] = stk[mid];
    	Build(L, mid, d ^ 1, nd[cur].ls);
    	Build(mid + 1, R, d ^ 1, nd[cur].rs);
    	pushup(cur);
    }
    inline bool che(int cur) {
    	int ls = nd[cur].ls;
    	return nd[ls].siz / nd[cur].siz >= alpha;
    }
    inline void Rebuild(int &cur) {
    	stop = 0;
    	Del(cur);
    	Build(1, stop, 0, cur);
    }
    void add(int d, int &cur) {
    	if (!cur) {
    		cur = ++ttot;
    		nd[cur] = tp;
    		return ;
    	}
    	if (tp.d[d] < nd[cur].d[d])	add(d ^ 1, nd[cur].ls);
    	else	add(d ^ 1, nd[cur].rs);
    	pushup(cur);
    	if (che(cur))	Rebuild(cur);
    }
    

    P5471 [NOI2019]弹跳

    又是卡空间,需要 K-D Tree 优化建图。不过还不行,不能显式地建出边,直接在 K-D Tree 的框架下模拟 Dijkstra 算法流程才行。不用剪枝可过 loj,需要剪枝才能过洛谷,剪枝也不能过 uoj。

    inline void Update(int cur, int v) {
    	if (v < dis[cur])
    		dis[cur] = v, q.push((Node){cur, dis[cur]});
    }
    void modify(int l, int r, int d, int u, int v, int cur) {
    	if (!cur || v >= dis[cur + n])	return ;
    	int al = nd[cur].mn[0], ar = nd[cur].mx[0], ad = nd[cur].mn[1], au = nd[cur].mx[1];
    	if (ar < l || al > r || au < d || ad > u)	return ;
    	if (l <= al && ar <= r && d <= ad && au <= u)	return Update(cur + n, v), void();
    	int x = nd[cur].d[0], y = nd[cur].d[1];
    	if (l <= x && x <= r && d <= y && y <= u)	Update(cur, v);
    	modify(l, r, d, u, v, nd[cur].ls);
    	modify(l, r, d, u, v, nd[cur].rs);
    }
    int mp[N];
    inline void dij() {
    	memset(dis, 0x3f, sizeof(dis));
    	dis[mp[1]] = 0;
    	q.push((Node){mp[1], 0});//Attention!!
    	while (!q.empty()) {
    		Node Nd = q.top(); q.pop();
    		int cur = Nd.cur;
    		if (vis[cur])	continue;
    		vis[cur] = true;
    		if (cur <= n) {
    			for (register unsigned int i = 0; i < mt[cur].size(); ++i) {
    				matrix mat = mt[cur][i];
    				modify(mat.l, mat.r, mat.d, mat.u, dis[cur] + mat.t, root);
    			}
    		} else {
    			cur -= n;
    			node nod = nd[cur];
    			int ls = nod.ls, rs = nod.rs;
    			Update(cur, dis[cur + n]);
    			if (ls)	Update(ls + n, dis[cur + n]);//Attention!!!
    			if (rs)	Update(rs + n, dis[cur + n]);//Attention!!!
    		}
    	}
    }
    

    注意

    • 那个 nth_element 的顺序是:左 + 1,中 + 1,右 + 1,cmp。注意加一(尽管之前有些代码没加一也能过)

    • 由于排序,建完树后每个节点的下标和原标号有所不同。手写 map 映射一下即可。

  • 相关阅读:
    《程序员修炼之道》——第二章 注重实效的途径(三)
    《程序员修炼之道》——第二章 注重实效的途径(二)
    《程序员修炼之道》——第二章 注重实效的途径(一)
    win10 磁盘占用高--- 禁用用户改善反馈 CompatTelRunner.exe
    ffmpeg拼接多个音频
    词云-wordcloud
    大数据指数日常应用
    搜索过滤Tip : title,site(搜标题和搜网站)
    eclipse下查看java源码设置
    sqlplus sys as sysdba
  • 原文地址:https://www.cnblogs.com/JiaZP/p/13367098.html
Copyright © 2011-2022 走看看