最近点对问题搞不懂?一篇文章就够了

2024-09-03 21:36

本文主要是介绍最近点对问题搞不懂?一篇文章就够了,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

标题:你以为找最近点对只是暴力计算?不,分治算法才是真正的王牌!

你以为最近点对问题就是简单的“比比看谁最近”?但你知道吗,很多人用了暴力解法,认为两两比较再选出最小距离就行了!表面上看,这确实是最直接的思路,但你若是真这么做,恭喜你,成功入坑!这种 (O(n^2)) 的算法可不是你想要的,尤其是在竞赛场上,那简直是拖后腿!在今天的这篇文章里,我们要来打破这个误解,揭开真正高效解决最近点对问题的神秘算法——分治法(Divide and Conquer)

为什么暴力法行不通?看清背后的时间复杂度!

最近点对问题很简单,给定一组点,要找到两个最近的点对。这不就是在点与点之间比距离吗?但如果你这么想,那就要小心了!暴力法的时间复杂度是 (O(n^2)),因为要比较每一对点的距离。这种方法在点数少时看起来还不错,但一旦数据量大了,它的效率就让你哭都哭不出来!但分治算法的出现,正是为了解决这一难题。

1. 分治法登场:把“大象”劈开来,效率提升一百倍!

分治法的思路是什么?一言以蔽之:分而治之!就像你切西瓜一样,把问题一分为二,先解决每一部分的小问题,再合并结果。分治法能将时间复杂度降到 (O(n \log n)),这可不是简单的一点点优化,而是质的飞跃!到底是怎么做到的呢?我们一步步来看!

步骤拆解:简单五步,步步为营
  1. 排序:从简单入手
    先将所有点按照 x 坐标进行排序(如果有必要,再按照 y 坐标排序)。你可能觉得这跟“比距离”没什么关系,但别急,高手的操作就在这些细节中!

  2. 分治分割:分而治之的第一步,开刀!
    将点集一分为二,分别成为左右两部分。注意了,这里选择的是中位数点,将整个点集分成几乎相等的两部分,这样才能确保后续算法的平衡性!

  3. 递归计算:子问题的解决方案,效率拉满!
    对于左右两部分,分别递归地求解最近点对。别小看这一步,这就是分治法的精髓所在!递归地处理子问题,解决了就可以合并了。

  4. 合并结果:跨区域比较是关键!
    两部分的最小距离可能不在同一部分内,而是跨越中线的。因此,我们需要在左右两部分之间进行一次合并检查,找出左右最近点之间的最小距离。这一步就考验你的理解能力了!此处只需考虑到距离中线不超过 (\delta) 的点,因为更远的就不可能更近。

  5. 巧妙的带宽问题:别让繁琐计算拖慢了你!
    你可能以为跨区域的比较很麻烦,但实际上,我们只需要考虑每个点最多 6 个候选点就行了!为什么?想象一下,这些点在一个 2(\delta) x (\delta) 的矩形区域内,且相隔距离不超过 (\delta)。巧妙的数学证明告诉你,这样的比较效率完全不需要担心。

看了这五步,你可能觉得这听起来依旧很复杂。别急,我们把所有步骤都放进一段代码中,真相就清晰了!

#include <stdio.h>
#include <stdlib.h>
#include <float.h>
#include <math.h>typedef struct {double x, y;
} Point;// 按x坐标排序
int compareX(const void* a, const void* b) {Point* p1 = (Point*)a;Point* p2 = (Point*)b;return (p1->x - p2->x);
}// 按y坐标排序
int compareY(const void* a, const void* b) {Point* p1 = (Point*)a;Point* p2 = (Point*)b;return (p1->y - p2->y);
}// 计算两点之间的欧几里得距离
double distance(Point p1, Point p2) {return sqrt((p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y));
}// 跨越分割线寻找最小距离
double stripClosest(Point strip[], int size, double d) {double min = d;  // 初始化最小距离为 dqsort(strip, size, sizeof(Point), compareY);  // 按 y 坐标排序for (int i = 0; i < size; ++i) {for (int j = i + 1; j < size && (strip[j].y - strip[i].y) < min; ++j) {double dist = distance(strip[i], strip[j]);if (dist < min) {min = dist;}}}return min;
}// 分治法计算最近点对
double closestUtil(Point points[], int n) {if (n <= 3) {double minDist = FLT_MAX;for (int i = 0; i < n; i++) {for (int j = i + 1; j < n; j++) {double dist = distance(points[i], points[j]);if (dist < minDist) {minDist = dist;}}}return minDist;}int mid = n / 2;Point midPoint = points[mid];double dl = closestUtil(points, mid);double dr = closestUtil(points + mid, n - mid);double d = fmin(dl, dr);Point strip[n];int j = 0;for (int i = 0; i < n; i++) {if (fabs(points[i].x - midPoint.x) < d) {strip[j] = points[i];j++;}}return fmin(d, stripClosest(strip, j, d));
}// 最近点对主函数
double closest(Point points[], int n) {qsort(points, n, sizeof(Point), compareX);return closestUtil(points, n);
}// 测试用例
int main() {Point points[] = {{2, 3}, {12, 30}, {40, 50}, {5, 1}, {12, 10}, {3, 4}};int n = sizeof(points) / sizeof(points[0]);printf("最近点对的距离: %.6f\n", closest(points, n));return 0;
}

看到了吗?当你掌握了分治法的精髓后,原本看似复杂的步骤就变得有条不紊。这种算法的美感和效率上的提升,可以说是计算几何中的一大经典了。

为什么分治法如此强大?因为它的细节优化无懈可击!

很多人都认为算法的效率提升就是简单的减少计算步骤,但分治法更进一步:它通过数学上的严谨证明,让每一个步骤都没有多余的操作。尤其是在跨区域合并的步骤,很多人会觉得“这一步复杂”,但实际上通过巧妙的数学推导,你会发现这个问题其实变得非常简单!

总结:别再被暴力法给坑了,掌握分治法才是你登顶的钥匙!

最近点对问题不仅仅是计算几何的一个小练习,它揭示了算法优化的深层逻辑。想要在竞赛中拔得头筹?想要在面试中成为闪亮的那颗星?那么你就需要掌握这样的算法,理解它的精髓,用它的威力来打破常规的认知。

还等什么?今天的分享就到这里,赶快动手实践起来吧!我们下次见!

这篇关于最近点对问题搞不懂?一篇文章就够了的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1134112

相关文章

mybatis和mybatis-plus设置值为null不起作用问题及解决

《mybatis和mybatis-plus设置值为null不起作用问题及解决》Mybatis-Plus的FieldStrategy主要用于控制新增、更新和查询时对空值的处理策略,通过配置不同的策略类型... 目录MyBATis-plusFieldStrategy作用FieldStrategy类型每种策略的作

linux下多个硬盘划分到同一挂载点问题

《linux下多个硬盘划分到同一挂载点问题》在Linux系统中,将多个硬盘划分到同一挂载点需要通过逻辑卷管理(LVM)来实现,首先,需要将物理存储设备(如硬盘分区)创建为物理卷,然后,将这些物理卷组成... 目录linux下多个硬盘划分到同一挂载点需要明确的几个概念硬盘插上默认的是非lvm总结Linux下多

Python Jupyter Notebook导包报错问题及解决

《PythonJupyterNotebook导包报错问题及解决》在conda环境中安装包后,JupyterNotebook导入时出现ImportError,可能是由于包版本不对应或版本太高,解决方... 目录问题解决方法重新安装Jupyter NoteBook 更改Kernel总结问题在conda上安装了

pip install jupyterlab失败的原因问题及探索

《pipinstalljupyterlab失败的原因问题及探索》在学习Yolo模型时,尝试安装JupyterLab但遇到错误,错误提示缺少Rust和Cargo编译环境,因为pywinpty包需要它... 目录背景问题解决方案总结背景最近在学习Yolo模型,然后其中要下载jupyter(有点LSVmu像一个

解决jupyterLab打开后出现Config option `template_path`not recognized by `ExporterCollapsibleHeadings`问题

《解决jupyterLab打开后出现Configoption`template_path`notrecognizedby`ExporterCollapsibleHeadings`问题》在Ju... 目录jupyterLab打开后出现“templandroidate_path”相关问题这是 tensorflo

如何解决Pycharm编辑内容时有光标的问题

《如何解决Pycharm编辑内容时有光标的问题》文章介绍了如何在PyCharm中配置VimEmulator插件,包括检查插件是否已安装、下载插件以及安装IdeaVim插件的步骤... 目录Pycharm编辑内容时有光标1.如果Vim Emulator前面有对勾2.www.chinasem.cn如果tools工

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

Java多线程父线程向子线程传值问题及解决

《Java多线程父线程向子线程传值问题及解决》文章总结了5种解决父子之间数据传递困扰的解决方案,包括ThreadLocal+TaskDecorator、UserUtils、CustomTaskDeco... 目录1 背景2 ThreadLocal+TaskDecorator3 RequestContextH

关于Spring @Bean 相同加载顺序不同结果不同的问题记录

《关于Spring@Bean相同加载顺序不同结果不同的问题记录》本文主要探讨了在Spring5.1.3.RELEASE版本下,当有两个全注解类定义相同类型的Bean时,由于加载顺序不同,最终生成的... 目录问题说明测试输出1测试输出2@Bean注解的BeanDefiChina编程nition加入时机总结问题说明

关于最长递增子序列问题概述

《关于最长递增子序列问题概述》本文详细介绍了最长递增子序列问题的定义及两种优化解法:贪心+二分查找和动态规划+状态压缩,贪心+二分查找时间复杂度为O(nlogn),通过维护一个有序的“尾巴”数组来高效... 一、最长递增子序列问题概述1. 问题定义给定一个整数序列,例如 nums = [10, 9, 2