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

Total Submission(s): 3520    Accepted Submission(s): 1247

Problem Description
The course of Software Design and Development Practice is objectionable. ZLC is facing a serious problem .There are many points in K-dimensional space .Given a point. ZLC need to find out the closest m points. Euclidean distance is used as the distance metric
between two points. The Euclidean distance between points p and q is the length of the line segment connecting them.In Cartesian coordinates, if p = (p1, p2,..., pn) and q = (q1, q2,..., qn)
are two points in Euclidean n-space, then the distance from p to q, or from q to p is given by:




Can you help him solve this problem?
 
Input
In the first line of the text file .there are two non-negative integers n and K. They denote respectively: the number of points, 1 <= n <= 50000, and the number of Dimensions,1 <= K <= 5. In each of the following n lines there is written k integers, representing
the coordinates of a point. This followed by a line with one positive integer t, representing the number of queries,1 <= t <=10000.each query contains two lines. The k integers in the first line represent the given point. In the second line, there is one integer
m, the number of closest points you should find,1 <= m <=10. The absolute value of all the coordinates will not be more than 10000.

There are multiple test cases. Process to end of file.
 
Output
For each query, output m+1 lines:

The first line saying :”the closest m points are:” where m is the number of the points.

The following m lines representing m points ,in accordance with the order from near to far

It is guaranteed that the answer can only be formed in one ways. The distances from the given point to all the nearest m+1 points are different. That means input like this:

2 2

1 1

3 3

1

2 2

1

will not exist.
 
Sample Input
3 2
1 1
1 3
3 4
2
2 3
2
2 3
1
 
Sample Output
the closest 2 points are:
1 3
3 4
the closest 1 points are:
1 3
 
Author
HIT
 
Source
 

【题解】

kd-tree,虽然不是曼哈顿距离。但是还是能够从曼哈顿距离里面得到启发的。

本来的估价函数的原则是

如果在矩形内就返回0.这点不变。

然后在矩形外就返回到这个矩形的最小曼哈顿距离。现在改成欧几里得距离就可以了。

然后返回这个距离作为估价函数。

取估价函数较小的那个方向更新解就可以了。

前k小的话可以加一个队列维护。因为k最大为10,所以没必要写二分。

具体的看代码吧。

着重看一下估价函数就好。

//数据范围实际上是可以不用定义long long的。但是保险起见。

//k维的并没有什么可怕的。。就是for 0->1变成for 0->k-1

//然后要注意多个维的点的输出最后一个维的坐标后面不能多空格。

【代码】

#include <cstdio>
#include <algorithm>
#include <cstring> using namespace std; const int MAXN = 60000;
const int MAXK_TH = 20;
const long long INF = 4410000000000000000; struct point
{
long long d[5], mi_n[5], ma_x[5];
int l, r;
}; struct data2
{
long long dis, d[5];
}; int n, k, root, now, k_th;
point t[MAXN], p[MAXN], op;
long long ans;
data2 dl[MAXK_TH]; bool cmp(point a, point b)
{
if (a.d[now] < b.d[now])
return true;
return false;
} void push_up(int rt)
{
int l = t[rt].l, r = t[rt].r;
for (int i = 0; i <= k - 1; i++)
{
if (l)
{
t[rt].ma_x[i] = max(t[l].ma_x[i], t[rt].ma_x[i]);
t[rt].mi_n[i] = min(t[l].mi_n[i], t[rt].mi_n[i]);
}
if (r)
{
t[rt].ma_x[i] = max(t[r].ma_x[i], t[rt].ma_x[i]);
t[rt].mi_n[i] = min(t[r].mi_n[i], t[rt].mi_n[i]);
}
}
} int build(int begin, int end, int fx)
{
int m = (begin + end) >> 1;
now = fx;
nth_element(p + begin, p + m, p + end + 1, cmp);
t[m] = p[m];
for (int i = 0; i <= k - 1; i++)
t[m].ma_x[i] = t[m].mi_n[i] = t[m].d[i];
if (begin < m)
t[m].l = build(begin, m - 1, (fx + 1) % k);
if (m < end)
t[m].r = build(m + 1, end, (fx + 1) % k);
push_up(m);
return m;
} void input_data()
{
memset(t, 0, sizeof(t));
for (int i = 1; i <= n; i++)
for (int j = 0; j <= k - 1; j++)
scanf("%lld", &p[i].d[j]);
root = build(1, n, 0);
} long long get_dis(point a, point b)
{
long long temp = 0;
for (int i = 0; i <= k - 1; i++)
temp += (a.d[i] - b.d[i])*(a.d[i] - b.d[i]);
return temp;
} long long sqr(long long x)
{
return x*x;
} long long gujia_min(int rt) //某个子树的估价最小函数
{
long long temp = 0;
for (int i = 0; i <= k - 1; i++)
{
//min(sqr(t[rt].ma_x[i] - op.d[i]), sqr(t[rt].mi_n[i] - op.d[i]));
long long temp1 = op.d[i] - t[rt].ma_x[i];
long long temp2 = t[rt].mi_n[i] - op.d[i];
if (temp1 > 0)
temp += sqr(op.d[i] - t[rt].ma_x[i]);
if (temp2 > 0)
temp += sqr(t[rt].mi_n[i] - op.d[i]);
}
return temp;
} void query_min(int rt)
{
long long dis = get_dis(t[rt], op);
int tempk = k_th;
while (dl[tempk].dis > dis)
{
tempk--;
if (!tempk)
break;
}
if (tempk != k_th) //如果比第k小的还小 就往前找一个合适的位置放进去。
{//然后第k小的就被挤掉 更新了。
for (int i = k_th; i >= tempk + 2; i--)
dl[i] = dl[i - 1];
dl[tempk + 1].dis = dis;
for (int i = 0; i <= k - 1; i++)
dl[tempk + 1].d[i] = t[rt].d[i];
}
long long gl = INF, gr = INF;
int l = t[rt].l, r = t[rt].r;
if (l)
gl = gujia_min(l);
if (r)
gr = gujia_min(r);
if (gl < gr)
{
if (gl < dl[k_th].dis)
query_min(l);
if (gr < dl[k_th].dis)
query_min(r);
}
else
{
if (gr < dl[k_th].dis)
query_min(r);
if (gl < dl[k_th].dis)
query_min(l);
}
} void output_ans()
{
int t;
scanf("%d", &t);
while (t--)
{
for (int i = 0; i <= k - 1; i++)
scanf("%lld", &op.d[i]);
scanf("%d", &k_th);
for (int i = 0; i <= k_th; i++)
dl[i].dis = INF;
ans = INF;
query_min(root);
printf("the closest %d points are:\n", k_th);
for (int i = 1; i <= k_th; i++)
{
for (int j = 0; j <= k - 2; j++)
printf("%lld ", dl[i].d[j]);
printf("%lld\n", dl[i].d[k - 1]);//最后一维的坐标后面不能加空格。
}
}
} int main()
{
//freopen("F:\\rush.txt", "r", stdin);
while (~scanf("%d%d", &n, &k)) //一直好奇前面的波浪号一样的是啥
{
input_data();
output_ans();
}
return 0;
}

【35.43%】【hdu 4347】The Closest M Points的更多相关文章

  1. 【改革春风吹满地 HDU - 2036 】【计算几何-----利用叉积计算多边形的面积】

    利用叉积计算多边形的面积 我们都知道计算三角形的面积时可以用两个邻边对应向量积(叉积)的绝对值的一半表示,那么同样,对于多边形,我们可以以多边形上的一个点为源点,作过该点并且过多边形其他点中的某一个的 ...

  2. 【HDU 2255】奔小康赚大钱 (最佳二分匹配KM算法)

    奔小康赚大钱 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Subm ...

  3. 【二分】【最长上升子序列】HDU 5489 Removed Interval (2015 ACM/ICPC Asia Regional Hefei Online)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5489 题目大意: 一个N(N<=100000)个数的序列,要从中去掉相邻的L个数(去掉整个区间 ...

  4. 【贪心】【模拟】HDU 5491 The Next (2015 ACM/ICPC Asia Regional Hefei Online)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5491 题目大意: 一个数D(0<=D<231),求比D大的第一个满足:二进制下1个个数在 ...

  5. 【动态规划】【二分】【最长上升子序列】HDU 5773 The All-purpose Zero

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5773 题目大意: T组数据,n个数(n<=100000),求最长上升子序列长度(0可以替代任何 ...

  6. 【动态规划】【KMP】HDU 5763 Another Meaning

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5763 题目大意: T组数据,给两个字符串s1,s2(len<=100000),s2可以被解读成 ...

  7. 【归并排序】【逆序数】HDU 5775 Bubble Sort

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5775 题目大意: 冒泡排序的规则如下,一开始给定1~n的一个排列,求每个数字在排序过程中出现的最远端 ...

  8. 【中国剩余定理】【容斥原理】【快速乘法】【数论】HDU 5768 Lucky7

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5768 题目大意: T组数据,求L~R中满足:1.是7的倍数,2.对n个素数有 %pi!=ai  的数 ...

  9. 【规律】【贪心】【数学】HDU 5573 Binary Tree

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5573 题目大意: 从1走到第k层,下一层的数是上一层的数*2或者*2+1,可以选择加上或者减去走的数 ...

随机推荐

  1. 69.fprintf fscanf

    fprintf //从读文件中提取字符串到info1.user和info1.password中 fscanf(pfr, "%s%s", info1.user, info1.pass ...

  2. .NET Entity Framework入门操作

    Entity Framework是微软借鉴ORM思想开发自己的一个ORM框架. ORM就是将数据库表与实体对象(相当于三层中的Model类)相互映射的一种思想. 最大的优点就是非常方便的跨数据库平台. ...

  3. System and method for critical address space protection in a hypervisor environment

    A system and method in one embodiment includes modules for detecting an access attempt to a critical ...

  4. 洛谷 P1636 Einstein学画画

    P1636 Einstein学画画 题目描述 Einstein学起了画画, 此人比较懒--,他希望用最少的笔画画出一张画... 给定一个无向图,包含n 个顶点(编号1~n),m 条边,求最少用多少笔可 ...

  5. jquery ui 分页插件 传入后台的连个參数名

    參数名: page .rows page=int(request.form.get('page',1).encode('u8')) rows1=int(request.form.get('rows', ...

  6. vue-cli打包项目后,可以修改配置文件

    问题: 前端需要修改后台服务器地址url,写好的配置文件会在npm run build 后压缩在一起,传到运行的前端服务器上后,需要到前端打包的源码,找到url地址进行修改.如果不在打包的源码修改,则 ...

  7. [python]-类与对象-上

    [类]是一个函数包.类中可以放置函数和变量,然后类中的函数可以很方便的使用类中的变量 1.类的定义 2.类中方法的调用 在类中被定义的函数被称为类的[方法],描述的是这个类能做什么.我们使用类名.函数 ...

  8. cdn缓存

    1:缓存是什么? 首先.看看没有站点没有接入CDN时.用户浏览器与server是怎样交互的: 假设中间加上一层CDN,那么用户浏览器与server的交互例如以下: client浏览器先检查是否有本地缓 ...

  9. 每日技术总结:Yarn和Npm大PK

    今天想用npm安装vue-cli@2.9 npm install --global vue-cli@2.9 卡半天,安装不成功,清空缓存,换taobao源重来,还是一样. 无奈之下换yarn yarn ...

  10. UWP 新手教程1——UWP的前世今生

    文件夹 引言 设备族群 UI 和通用输入模式 通用控件和布局面板 工具 自适应扩展 通用输入处理 引言 在本篇文章中,可以掌握下面知识: 设备族群,怎样决定目标设备 新的UI控件和新面板帮助你适应不同 ...