KDTree模板,在m维空间中找最近的k个点,用的是欧几里德距离。

理解了好久,昨晚始终不明白那些“估价函数”,后来才知道分情况讨论,≤k还是=k,在当前这一维度距离过线还是不过线,过线则要继续搜索另一个子树。还有别忘了当前这个节点!

#include<cmath>
#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define read(x) x=getint()
using namespace std;
typedef long long LL;
const int N = 50003;
const int inf = 0x7fffffff;
int getint() {
int k = 0, fh = 1; char c = getchar();
for(; c < '0' || c > '9'; c = getchar())
if (c == '-') fh = -1;
for(; c >= '0' && c <= '9'; c = getchar())
k = k * 10 + c - '0';
return k * fh;
}
int n, m, root, D;
LL minn;
struct P {
int d[5], mx[5], mn[5], l, r, id;
P (): l(0), r(0), id(0) {};
int &operator [] (int x) {return d[x];}
bool operator < (P point) const {return d[D] < point[D];}
} T[N << 1], po[N];
priority_queue <pair <LL, int> > Q;
LL sqr(LL x) {return x * x;}
LL dis(P a, P b) {
LL ret = 0;
for(int i = 0; i < m; ++i) ret += sqr(a[i] - b[i]);
return ret;
}
void pushup(int x, int y) {
for(int i = 0; i < m; ++i)
T[x].mn[i] = min(T[x].mn[i], T[y].mn[i]),
T[x].mx[i] = max(T[x].mx[i], T[y].mx[i]);
}
int Build(int l, int r, int dd) {
D = dd; int mid = (l + r) >> 1; nth_element(po + l, po + mid, po + r + 1);
for(int i = 0; i < m; ++i)
T[mid].mn[i] = T[mid].mx[i] = T[mid].d[i] = po[mid].d[i];
T[mid].id = mid;
if (l < mid) T[mid].l = Build(l, mid - 1, (dd + 1) % m);
if (mid < r) T[mid].r = Build(mid + 1, r, (dd + 1) % m);
if (T[mid].l) pushup(mid, T[mid].l);
if (T[mid].r) pushup(mid, T[mid].r);
return mid;
}
void ask(int rt, int dd, P p, int k) {
int L = T[rt].l, R = T[rt].r;
if (p[dd] >= T[rt][dd]) swap(L, R);
if (L) ask(L, (dd + 1) % m, p, k);
bool pd = 0; LL di = dis(T[rt], p); minn = min(minn, di);
if (Q.size() < k) { Q.push(pair <LL, int> (di, rt)); pd = 1;}
else {
if (di < Q.top().first) Q.pop(), Q.push(make_pair(di, rt));
if (sqr(p[dd] - T[rt][dd]) < Q.top().first) pd = 1;
}
if (pd && R) ask(R, (dd + 1) % m, p, k);
}
int ans[N];
int main() {
while (~scanf("%d%d", &n, &m)) {
for(int i = 1; i <= n; ++i)
for(int j = 0; j < m; ++j)
read(po[i][j]);
memset(T, 0, sizeof(T));
root = Build(1, n, 0);
int qq; read(qq);
for(; qq; --qq) {
P p; int k;
for(int i = 0; i < m; ++i)
read(p[i]);
read(k);
printf("the closest %d points are:\n", k);
minn = inf; ask(root, 0, p, k);
while (!Q.empty()) {
ans[++ans[0]] = Q.top().second;
Q.pop();
}
for(; ans[0]; --ans[0])
for(int i = 0; i < m; ++i)
printf("%d%c", T[ans[ans[0]]][i]," \n"[i == m - 1]);
}
}
return 0;
}

我就是弱啊~~~

【BZOJ 3053】The Closest M Points的更多相关文章

  1. 【BZOJ】【3053】The Closest M Points

    KD-Tree 题目大意:K维空间内,与给定点欧几里得距离最近的 m 个点. KD树啊……还能怎样啊……然而扩展到k维其实并没多么复杂?除了我已经脑补不出建树过程……不过代码好像变化不大>_&g ...

  2. 【35.43%】【hdu 4347】The Closest M Points

    Time Limit: 16000/8000 MS (Java/Others)    Memory Limit: 98304/98304 K (Java/Others) Total Submissio ...

  3. 【BZOJ 1150】 1150: [CTSC2007]数据备份Backup (贪心+优先队列+双向链表)

    1150: [CTSC2007]数据备份Backup Description 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味 的,因此你想设 ...

  4. Kruskal算法及其类似原理的应用——【BZOJ 3654】tree&&【BZOJ 3624】[Apio2008]免费道路

    首先让我们来介绍Krukal算法,他是一种用来求解最小生成树问题的算法,首先把边按边权排序,然后贪心得从最小开始往大里取,只要那个边的两端点暂时还没有在一个联通块里,我们就把他相连,只要这个图里存在最 ...

  5. 【BZOJ 2957】楼房重建&&Codechef COT5 Count on a Treap&&【NOIP模拟赛】Weed 线段树的分治维护

    线段树是一种作用于静态区间上的数据结构,可以高效查询连续区间和单点,类似于一种静态的分治.他最迷人的地方在于“lazy标记”,对于lazy标记一般随我们从父区间进入子区间而下传,最终给到叶子节点,但还 ...

  6. LCA 【bzoj 4281】 [ONTAK2015]Związek Harcerstwa Bajtockiego

    [bzoj 4281] [ONTAK2015]Związek Harcerstwa Bajtockiego Description 给定一棵有n个点的无根树,相邻的点之间的距离为1,一开始你位于m点. ...

  7. 【hdu4347】The Closest M Points 【KD树模板】

    题意 一个k维空间,给出n个点的坐标,给出t个询问,每个询问给出一个点的坐标和一个m.对于每个询问找出跟这个点最接近的m个点 分析 kd树的模板题. #include <cstdio> # ...

  8. 【BZOJ 1191】 [Apio2010]特别行动队 (斜率优化)

    dsy1911: [Apio2010]特别行动队 [题目描述] 有n个数,分成连续的若干段,每段的分数为a*x^2+b*x+c(a,b,c是给出的常数),其中x为该段的各个数的和.求如何分才能使得各个 ...

  9. 【BZOJ 1096】 [ZJOI2007]仓库建设 (斜率优化)

    1096: [ZJOI2007]仓库建设 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3940  Solved: 1736 Description ...

随机推荐

  1. 《2016ThoughtWorks技术雷达峰会----微服务架构》

    微服务架构   王键,ThoughtWorks, 首席咨询师 首先微服务架构的定义,thoughtWorks在2012年3月的技术雷达中这样定义: “微服务架构是一种架构,它提倡将单一应用程序划分为一 ...

  2. POJ 1696 Space Ant --枚举,模拟,贪心,几何

    题意: 有很多点,从最右下角的点开始走起,初始方向水平向右,然后以后每步只能向左边走,问最多能走多少个点. 解法: 贪心的搞的话,肯定每次选左边的与它夹角最小的点,然后走过去. 然后就是相当于模拟地去 ...

  3. GitHub 下载文件夹

    工具 TortoiseSVN 步骤 1.打开要下载的项目,选中要下载的文件夹,右键 选择 复制链接地址 2.把链接中的 tree/master 改成 trunk ,(trunk是master分支,可以 ...

  4. quad 和 plane 区别是什么?

    Quad就是两个三角形组成四边形,Plane会有很多三角形,哦也 貌似Quad拖上去后看不见,很薄的感觉

  5. J2EE中关于tomcat的maxIdle、maxActive、maxActive相关配置

    一.基本概念 1 maxActive 连接池的最大数据库连接数.设为0表示无限制,一般把maxActive设置成可能的并发量就行了 2 maxIdle 最大的空闲连接数 3 maxWait 最大建立连 ...

  6. java 22 - 16 多线程之生产者和消费者的问题

    生产者和消费者问题的描述图 通过上图,我们可以发现: 生产者和消费者使用的都是同一个资源(肉包子) 所以,当使用线程的时候,这两类的锁也是同一把锁(为了避免出现线程安全问题) 例子:学生信息的录入和获 ...

  7. css3动画中的steps值详解

    css3的动画的animation-timing-function属性定义了动画的速度曲线,一般的速度曲线大家都知道,什么ease,linear,ease-in,ease-out,还有自定义贝塞尔曲线 ...

  8. (原创)mybatis学习二,spring和mybatis的融合

    mybatis学习一夯实基础 上文介绍了mybatis的相关知识,这一节主要来介绍mybaits和spring的融合 一,环境搭建 1,jar包下载,下载路径为jar包 2,将包导入到java工程中 ...

  9. FFT的物理意义

    来源:学步园 FFT(Fast Fourier Transform,快速傅立叶变换)是离散傅立叶变换的快速算法,也是我们在数字信号处理技术中经常会提到的一个概念.在大学的理工科课程中,在完成高等数学的 ...

  10. .net混淆、反编译工具调查

    常用的工具列表[比较常见的] 混淆器.加密 Dotfuscator VS默认带的工具,不过是个社区版 强度不大 dotNET Reactor 使用了NativeCode 和混淆的形式 Xenocode ...