使用优先队列找到离原点最近的K个点

使用优先队列找到离原点最近的k个点

在这个问题中,我们将从给定的 N 个点中找到 2D 平面中距离原点最近的 K 个点。

我们可以使用标准的欧氏距离公式来计算原点到每个给定点之间的距离。之后,我们可以将有距离的点存储到数组中,根据距离对数组进行排序,并取前K个点。

然而,我们还可以使用优先队列根据点与原点的距离来存储二维点。之后,我们可以从队列中出队K次。

问题陈述− 我们在二维平面上给定了N个点。我们需要找到离平面原点最近的K个点。

注意– 将欧几里得距离视为原点和平面给定点之间的距离。

示例

输入

points = {{2, -2}, {-5, 1}, {2, 1}, {0, 3}, {6, 0}, {5, 5}, {4, 9}}, K = 4

输出

{2,1} {2,-2} {0,3} {-5,1}

Explanation − 这里是每个点到原点的欧几里得距离。

(2, −2) −> sqrt(8)

(−5, 1) −> sqrt(26)

(2, 1) -> sqrt(5)

(0, 3) −> sqrt(9)

(6, 0) -> sqrt(36)

(5, 5) -> sqrt(50)

(4, 9) -> sqrt(97)

因此,4 个最近的点是 {2,1} {2,−2} {0,3} {−5,1}。

输入

points = {{1, 2}, {2, 1}, {-2, 1}, {1, -2}} K = 2

输出

{1, 2}, {2, 1}

Explanation − 所有点到原点的距离都是相同的。因此,我们可以将任意2个点作为输出打印。

输入

points = {{1, 3}, {6, 7}, {1, 1}, {1, 0}} K = 4

输出

{1, 0}, {1, 1}, {1, 3}, {6, 7}

解释− K 等于给定点。因此,我们需要打印所有点。

方法一

在这种方法中,我们将使用 sort() 方法对数组进行排序。此外,我们将使用比较器函数根据点与原点的距离对点进行排序。之后,我们打印排序数组的前 K 个元素。

算法

步骤 1 − 使用sort()方法对列表进行排序,并将distComparator()函数作为第三个参数传递。

第二步− 定义distComparator()函数来比较给定点的距离。该函数以p和q对作为参数。

步骤 2.1 − 获取点 p 到原点的距离并将其存储在 dist_p 中。

步骤 2.2 − 将点 q 到原点的距离存储在 dist_q 变量中。

步骤 2.3 − 如果 dist_p 小于 dist_q,则返回 true。否则,返回 false。

第 3 步– 遍历数组,并打印数组的前 K 个点。

示例

#include using namespace std;bool distComparator(const pair &p, const pair &q) {    int dist_p = p.first * p.first + p.second * p.second;    int dist_q = q.first * q.first + q.second * q.second;    return dist_p < dist_q;}vector<pair> findKPoints(vector<pair> points, int n, int k) {    vector<pair> res_points;    sort(points.begin(), points.end(), distComparator);    // Get the first K points    for (int p = 0; p < k; p++)     {        res_points.push_back({points[p].first, points[p].second});    }    return res_points;}int main() {    int k = 4, n = 7;    vector<pair> points{{2, -2}, {-5, 1}, {2, 1}, {0, 3}, {6, 0}, {5, 5}, {4, 9}};    vector<pair> res = findKPoints(points, n, k);    cout << "The " << k << " closest points from origin are - ";    for (int p = 0; p < k; p++) {        cout << "{" << res[p].first << "," << res[p].second << "} ";    }    return 0;}

输出

The 4 closest points from origin are - {2,1} {2,-2} {0,3} {-5,1}

时间复杂度 – 对数组进行排序的时间复杂度为O(NlogN)。

空间复杂度 – O(N) 用于对数组进行排序。

方法二

在这种方法中,我们将使用优先级队列来插入点。此外,我们将使用比较器函数和优先级队列来根据距原点的最短距离来存储点。

算法

步骤 1 − 定义‘res_points’列表,用于存储K个最近的点。

步骤 2– 定义优先级队列。这里,‘pair’表示使用优先级队列来存储整数对。 ‘vector>’表示使用向量来存储对。此外,我们还使用了带有优先级队列的“cmp”函数。

第三步− 定义cmp()函数来比较两个点到原点的欧几里得距离。

步骤 3.1 – 根据 a 点到原点的距离大于 b 点到原点的距离,返回布尔值。

第 4 步– 将数组的每个元素插入优先级队列。

第5步− 从队列中弹出前K个元素,并将它们存储在res_points列表中。

第 6 步− 返回点的 res_points 列表。

示例

#include using namespace std;vector<pair> findKPoints(vector<pair> points, int n, int k) {    vector<pair> res_points;    // Custom comparator to compare points based on their distance from the origin    auto cmp = [](const pair& a, const pair& b) {        return (a.first * a.first + a.second * a.second) > (b.first * b.first + b.second * b.second);    };    // Use the custom comparator in the priority_queue    priority_queue<pair, vector<pair>, decltype(cmp)> p_que(cmp);    for (int p = 0; p < n; p++) {        // Inserting points in a queue        p_que.push(points[p]);    }    // Get first K points    while (k--) {        auto temp = p_que.top();        res_points.push_back(temp);        p_que.pop();    }    return res_points;}int main() {    int k = 4, n = 7;    vector<pair> points{{2, -2}, {-5, 1}, {2, 1}, {0, 3}, {6, 0}, {5, 5}, {4, 9}};    vector<pair> res = findKPoints(points, n, k);    cout << "The " << k << " closest points from origin are - ";    for (int p = 0; p < k; p++) {        cout << "{" << res[p].first << "," << res[p].second << "} ";    }    return 0;}

输出

The 4 closest points from origin are - {2,1} {2,-2} {0,3} {-5,1}

时间复杂度 – O(N*logN) 将 N 个元素插入优先级队列。这里,N是总点数。

空间复杂度 – 在优先级队列中存储点的 O(N)。

优先队列使用堆数据结构。因此,插入和删除元素只需O(logN)的时间。这两种方法都需要相同的内存和时间。然而,第二种方法更高效,因为它使用了堆数据结构。

以上就是使用优先队列找到离原点最近的K个点的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1445208.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 22:17:57
下一篇 2025年12月12日 15:15:01

相关推荐

  • 检查是否可能从原点到达给定圆的周长上的任意点

    圆的周长可以定义为圆的外边界。它是圆的周长。圆周围的每个点都遵循某些属性,如下所示 – 点 (x,y) 位于圆内,使得 $mathrm{x^2 + y^2 点 (x,y) 位于圆上,使得 $mathrm{x^2 + y^2 = R^2}$ 点 (x,y) 位于圆外,使得 $mathrm{…

    2025年12月17日
    000
  • Golang如何使用container/heap实现优先队列

    答案:Go语言通过container/heap包实现优先队列,需自定义类型并实现heap.Interface接口的五个方法;其中Len、Less、Swap为值接收者,Push和Pop为指针接收者;通过heap.Init初始化堆,heap.Push和heap.Pop进行入队出队操作;示例中以prior…

    好文分享 2025年12月16日
    000
  • Python中的堆和优先队列的使用场景有哪些?

    Python中的堆和优先队列的使用场景有哪些? 堆是一种特殊的二叉树结构,常用于高效地维护一个动态的集合。Python中的heapq模块提供了堆的实现,可以方便地进行堆的操作。 优先队列也是一种特殊的数据结构,不同于普通的队列,它的每个元素都有一个与之相关的优先级。最高优先级的元素先被取出。Pyth…

    2025年12月13日
    000
  • Python中的堆和优先队列是如何实现的?

    Python中的堆和优先队列是如何实现的? 堆和优先队列是在计算机科学中常用的数据结构。在Python中,我们可以使用heapq模块来实现堆和优先队列。 堆是一种特殊的完全二叉树,在堆中,每个父节点的值都比它的子节点的值要小(或大),这样的堆被称为小根堆(或大根堆)。在Python中,堆可以通过列表…

    2025年12月13日
    000
  • PHP中如何实现数组优先队列?

    在php中实现数组优先队列可以使用splpriorityqueue类。1) 使用splpriorityqueue类创建优先队列。2) 通过insert方法添加元素,优先级高的元素排在前面。3) 可以设置比较策略以改变相同优先级元素的排序行为。4) 注意性能瓶颈、优先级冲突和序列化问题。5) 可以通过…

    2025年12月10日
    000
  • Golang如何使用container/heap实现优先队列

    答案:Go语言通过container/heap包实现优先队列,需自定义类型并实现heap.Interface接口的五个方法;其中Len、Less、Swap为值接收者,Push和Pop为指针接收者;通过heap.Init初始化堆,heap.Push和heap.Pop进行入队出队操作;示例中以prior…

    2025年12月2日 后端开发
    000
  • 在Java中如何使用PriorityQueue实现优先队列_PriorityQueue操作指南

    PriorityQueue基于堆实现,默认最小堆,poll()返回最小值;通过Comparator可实现最大堆或自定义排序,常用于任务调度、Dijkstra等场景。 在Java中,PriorityQueue 是实现优先队列最常用的方式。它基于堆(heap)数据结构,默认使用最小堆,也就是说队列头部的…

    2025年12月2日 java
    000
  • PHP数据结构:优先队列的应用,掌控有序元素的获取

    优先队列允许按优先级存储和访问元素,基于可比较标准(如值、时间戳或自定义逻辑)设定优先级。php 中的实现方法包括 splpriorityqueue 类和 min/max 堆。实战案例演示了如何使用 splpriorityqueue 类创建优先队列并按优先级获取元素。 PHP 数据结构:优先队列的应…

    2025年11月9日 后端开发
    000

发表回复

登录后才能评论
关注微信