escription

  平面上有n个点。现在有m次询问,每次给定一个点(px, py)和一个整数k,输出n个点中离(px, py)的距离第k大的点的标号。如果有两个(或多个)点距离(px, py)相同,那么认为标号较小的点距离较大。

Input

  第一行,一个整数n,表示点的个数。
  下面n行,每行两个整数x_i, y_i,表示n个点的坐标。点的标号按照输入顺序,分别为1..n。
  下面一行,一个整数m,表示询问个数。
  下面m行,每行三个整数px_i, py_i, k_i,表示一个询问。

Output

  m行,每行一个整数,表示相应的询问的答案。
题解:  和之前一道题几乎一模一样,直接用堆 + $KDtree$ 维护即可.
#include<bits/stdc++.h>
#define maxn 200000
#define inf 1000000000000000
#define mid ((l+r)>>1)
#define rson(x) (t[x].ch[1])
#define lson(x) (t[x].ch[0])
#define ll long long
using namespace std;
void setIO(string s)
{
string in = s + ".in";
freopen(in.c_str(), "r" , stdin);
}
int d, n, m;
struct Node
{
ll dis;
int k;
Node (ll dis = 0, int k = 0) : dis(dis), k(k) {}
bool operator < (Node b) const
{
return dis == b.dis ? b.k > k : b.dis < dis;
}
};
priority_queue <Node> Q;
// return if a is less than b
bool cmp1(Node a, Node b)
{
return a.dis == b.dis ? a.k > b.k : a.dis < b.dis;
}
ll sqr(ll a)
{
return a * a;
}
struct ND
{
int ch[2], id;
ll p[2], minv[2], maxv[2];
}t[maxn],T;
bool cmp(ND a, ND b)
{
return a.p[d] == b.p[d] ? a.p[d ^ 1] < b.p[d ^ 1] : a.p[d] < b.p[d];
}
void pushup(int x, int y)
{
for(int i = 0; i < 2 ; ++i)
{
t[x].minv[i] = min(t[x].minv[i], t[y].minv[i]);
t[x].maxv[i] = max(t[x].maxv[i], t[y].maxv[i]);
}
}
int build(int l, int r, int o)
{
d = o;
nth_element(t + l, t + mid, t + 1 + r, cmp);
for(int i = 0; i < 2 ; ++i)
t[mid].minv[i] = t[mid].maxv[i] = t[mid].p[i];
lson(mid) = rson(mid) = 0;
if(mid > l)
{
lson(mid) = build(l, mid - 1, o ^ 1);
pushup(mid, lson(mid));
}
if(r > mid)
{
rson(mid) = build(mid + 1, r, o ^ 1);
pushup(mid, rson(mid));
}
return mid;
}
ll getmax(int x)
{
ll ans = 0;
for(int i = 0; i < 2 ; ++i)
{
ans += max(sqr(t[x].minv[i] - T.p[i]), sqr(t[x].maxv[i] - T.p[i]));
}
return ans;
}
void query(int x, ll x1, ll y1)
{
ll cur = sqr(t[x].p[0] - x1) + sqr(t[x].p[1] - y1), dl, dr, dn = getmax(x);
if(dn < Q.top().dis) return;
if(cmp1(Q.top(), Node(cur, t[x].id)))
{
Q.pop();
Q.push(Node(cur, t[x].id));
}
dl = lson(x) ? getmax(lson(x)) : -inf;
dr = rson(x) ? getmax(rson(x)) : -inf;
if(dl > dr)
{
if(dl >= Q.top().dis) query(lson(x), x1, y1);
if(dr >= Q.top().dis) query(rson(x), x1, y1);
}
else
{
if(dr >= Q.top().dis) query(rson(x), x1, y1);
if(dl >= Q.top().dis) query(lson(x), x1, y1);
}
}
int main()
{
int i, j, k, root;
ll x, y;
// setIO("input");
scanf("%d",&n);
for(i = 1; i <= n ; ++i) scanf("%lld%lld",&t[i].p[0],&t[i].p[1]), t[i].id = i;
root = build(1, n, 0);
scanf("%d",&m);
while(m --)
{
scanf("%lld%lld%d",&x,&y,&k);
while(!Q.empty()) Q.pop();
for(i = 1; i <= k ; ++i) Q.push(Node(-2333, -1));
T.p[0] = x, T.p[1] = y;
query(root, x, y);
printf("%d\n",Q.top().k);
}
return 0;
}

  

BZOJ 2626: JZPFAR KDtree + 堆的更多相关文章

  1. BZOJ 2626 JZPFAR(KD-tree)

    题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=2626 题意:平面上有n个点.现在有m次询问,每次给定一个点(px, py)和一个整数k, ...

  2. bzoj 2626: JZPFAR【KD-tree】

    和3053差不多,把pair first做成负数就可以用大根堆维护了 注意:要开long long:比较的时候因为编号也占权重所以要比较pair:编号不是mid!不是mid!是初始输入的那个编号!搞混 ...

  3. 【BZOJ2626】JZPFAR kd-tree+堆

    [BZOJ2626]JZPFAR Description 平面上有n个点.现在有m次询问,每次给定一个点(px, py)和一个整数k,输出n个点中离(px, py)的距离第k大的点的标号.如果有两个( ...

  4. bzoj 2626: JZPFAR k-D树

    题目大意: 平面上n个点,每次给出一个点,求这个点的k远点 题解: 什么叫做k远点呢... 1 2 3 4 5中5是第一远,4是第二远... 看来我语文学的不好 那么我们直接上k-D Tree求k邻近 ...

  5. BZOJ 2626 & KDtree

    题意: 二维平面n个点 每次给出一个点询问距离第k小的点. SOL: kdtree裸题,抄了一发别人的模板...二维割起来还是非常显然的.膜rzz的论文. 不多说了吧.... Code: /*==== ...

  6. 【24.91】【Tsinsen 1302】&【BZOJ 2626】JZPFAR

    时间限制:5.0s   内存限制:256.0MB   总提交次数:547   AC次数:137   平均分:40.31 将本题分享到:        查看未格式化的试题   提交   试题讨论 试题来 ...

  7. 【BZOJ-4520】K远点对 KD-Tree + 堆

    4520: [Cqoi2016]K远点对 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 490  Solved: 237[Submit][Status ...

  8. BZOJ4520 CQOI2016K远点对(KD-Tree+堆)

    堆维护第k大,每个点KD-Tree上A*式查询较远点,跑得飞快,复杂度玄学. #include<iostream> #include<cstdio> #include<c ...

  9. 【BZOJ4520】[Cqoi2016]K远点对 kd-tree+堆

    [BZOJ4520][Cqoi2016]K远点对 Description 已知平面内 N 个点的坐标,求欧氏距离下的第 K 远点对. Input 输入文件第一行为用空格隔开的两个整数 N, K.接下来 ...

随机推荐

  1. Python网络爬虫(一):初步认识网络爬虫

    不管你是因为什么原因想做一个网络爬虫,首先做的第一件事情就是要了解它. 在了解网络爬虫之前一定要牢记下面4个要点,这是做网络爬虫的基础: 1.抓取 py的urllib不一定去用.可是要学.假设你还没用 ...

  2. Solid Edge如何制作爆炸图

    1 最方便的方式是自动爆炸 点击应用程式-"爆炸-涂彩-动画"   选择要爆炸的对象(默认是顶层组立件),要爆炸的距离(默认系统根据零件大小自动生成距离,你也可以手动设定距离)即可 ...

  3. iOS: 学习笔记, Swift操作符定义

    Swift操作符能够自行定义, 仅仅须要加上简单的标志符就可以. @infix 中置运算. 如+,-,*,/运算 @prefix 前置运算. 如- @postfix 后置运算. a++, a-- @a ...

  4. Pell Sequence

    /* * PellSequence.cpp * * Created on: 2013-09-08 16:46 * Author: lg * Description: a1 = 1, a2 = 2, . ...

  5. BNUOJ 13098 约瑟夫环问题

    C. Josephus Problem 题目链接:http://www.bnuoj.com/v3/contest_show.php?cid=7095#problem/C 题目描述   The hist ...

  6. 【OI】同余方程

    一.同余方程的判定 我们知道同余方程是形如 ax ≡ b (mod n)   的东西,用文字表达就是: ax和b除以n的余数相同 那么,经过如下推理:(用=代替恒等于) ax=b (mod n) ax ...

  7. J - 玩游戏

    小A和小B玩游戏,初始的时候小A给小B一组包含n个数的数组.他们按如下的规则进行: 每次小B得到一组数,他把这组数的和加到自己的分数里面(他的初始分数是0),然后他把这组数还给小A. 如果小A得到的这 ...

  8. linux内核中的宏ffs(x)

    linux内核中ffs(x)宏是平台相关的宏,在arm平台,该宏定义在 arch/arm/include/asm/bitops.h #define ffs(x) ({ unsigned long __ ...

  9. kentico9开始移除的webpart

    https://devnet.kentico.com/articles/fighting-featuritis https://blog.intercom.com/product-strategy-m ...

  10. 在word中doc与docx的区别是什么(整理)

    在word中doc与docx的区别是什么(整理) docx 是Office2007使用的,是用新的基于XML的压缩文件格式取代了其目前专有的默认文件格式,在传统的文件名扩展名后面添加了字母x(即.do ...