【蛮力算法和分治法】平面最接近点对(Python)

2024-01-09 21:04

本文主要是介绍【蛮力算法和分治法】平面最接近点对(Python),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

问题介绍

        随机产生平面若干点,利用蛮力算法和分治算法找到平面的最接近点对,并考查随 n 变大时,两者的效率差异、实验效率和理论效率的一致性。平面点集能直观的进行观察

问题分析

        蛮力法很好理解,遍历所有的点计算其中所有的点之间的距离,并设置一个最小距离数,持续遍历时持续和结果比较,得到最小的距离。

        分治算法的思想则是,将大问题分解成小问题分别求解后合并,此处就是如此,将点所在的区域持续的分隔成左右两部分,分别求解,当区域不断被划分,区域中只剩下三个点的时候,采用蛮力法对点之间的距离进行计算。

        这样算出来的距离只是其中一半,另一半也如此,随后进行比较得到最小距离,但别忘了,要是距离最短的两个点,一个在左半部分一个在右半部分,那该怎么办?这就是我们的“跨越中线问题”。

        中线问题解决思路则是,当左右两边的最小距离(min_distance)求出后,往中线左右各取最小距离范围,即中间带宽的距离为2 * min_distance。之所以是这样的结果,原因是这样能够保证可以取到最小结果的点被包含在带宽而免去其它多余的计算。如果按照这个取,当中间线上有点,这个点和其他的在同一条水平线上的点的距离如果在这个范围,必然比min_distance小,这样的点才有计算的价值,否则必然距离比min_distance大的点放进来只会增加计算量且无用。

        时间复杂度上,最开始对点集进行排序,时间复杂度为O(nlogn)。随后的分隔工作采用二分法,假设总的点数量为n,分隔区域的次数为k次,那么2 ^ k = n,则k = log2n。每次分隔后都需要完成合并操作,合并最后的结果比较得出最小值,所以时间复杂度为O(n)。该算法的总时间复杂度为O(nlogn)。

        对图的展示采用python中的包,同时,既然要实验观察两者的各个效率,那就需要观察其运行时间,放入多点进去,观察两者时间上的差别。

代码

        最终可以得到四不同n的四个图,图中的最近点对会用红色的线连接,以及两个算法解决同样数量的点集的时间分别为多少。

import time
import random
import mathfrom matplotlib import pyplot as plt# 生成随机点集
def generate_random_points(n):points = [(random.random(), random.random()) for _ in range(n)]return points# 计算两点之间的距离
def distance(p1, p2):return math.sqrt((p1[0] - p2[0]) ** 2 + (p1[1] - p2[1]) ** 2)# 暴力求解最近点对的距离
def brute_force(points):min_distance = float('inf')for i in range(len(points)):for j in range(i + 1, len(points)):d = distance(points[i], points[j])if d < min_distance:min_distance = dreturn min_distance# 最近点对问题的分治算法
def closest_pair(points):# 根据 x 坐标排序sorted_points = sorted(points, key=lambda x: x[0])return closest_pair_recursion(sorted_points)def closest_pair_recursion(points):n = len(points)if n <= 3:return brute_force(points)mid = n // 2mid_point = points[mid]# 取左右部分left_points = points[:mid]right_points = points[mid:]# 递归处理左右半部分的最近点对距离left_min = closest_pair_recursion(left_points)right_min = closest_pair_recursion(right_points)min_distance = min(left_min, right_min)strip_points = [point for point in points if abs(point[0] - mid_point[0]) < min_distance]  # 取中间带宽内的点strip_min = strip_closest(strip_points, min_distance)  # 处理中间带宽内的最近点对距离return min(min_distance, strip_min)def strip_closest(strip_points, min_distance):min_distance_strip = min_distancen = len(strip_points)  # 获取中间带宽内的点的数量n。for i in range(n):j = i + 1# j小于中间带宽内的点的数量n,并且当前点与下一个点之间的纵坐标差小于最小距离min_distance_strip。while j < n and (strip_points[j][1] - strip_points[i][1]) < min_distance_strip:min_distance_strip = min(min_distance_strip, distance(strip_points[i], strip_points[j]))j += 1return min_distance_strip# 主函数
def main():ns = [10, 100, 1000, 10000]for n in ns:print("当n为", n)points = generate_random_points(n)startTime = time.time()minDistanceBruteForce = brute_force(points)endTime = time.time()timeTaken = endTime - startTimeprint("Brute force time:", timeTaken)print("Minimum distance (brute force):", minDistanceBruteForce)startTime = time.time()minDistanceClosestPair = closest_pair(points)endTime = time.time()timeTaken = endTime - startTimeprint("Closest pair time:", timeTaken)print("Minimum distance (closest pair):", minDistanceClosestPair)# 画出点集x = [point[0] for point in points]y = [point[1] for point in points]plt.figure()plt.scatter(x, y, color='b', label='Points')# 标记最近点对for i in range(len(points)):for j in range(i + 1, len(points)):if distance(points[i], points[j]) == minDistanceClosestPair:plt.plot([points[i][0], points[j][0]], [points[i][1], points[j][1]], color='r')plt.title('Random Points')plt.legend()plt.show()# 执行主函数
if __name__ == "__main__":main()

这篇关于【蛮力算法和分治法】平面最接近点对(Python)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python基础文件操作方法超详细讲解(详解版)

《Python基础文件操作方法超详细讲解(详解版)》文件就是操作系统为用户或应用程序提供的一个读写硬盘的虚拟单位,文件的核心操作就是读和写,:本文主要介绍Python基础文件操作方法超详细讲解的相... 目录一、文件操作1. 文件打开与关闭1.1 打开文件1.2 关闭文件2. 访问模式及说明二、文件读写1.

Python将博客内容html导出为Markdown格式

《Python将博客内容html导出为Markdown格式》Python将博客内容html导出为Markdown格式,通过博客url地址抓取文章,分析并提取出文章标题和内容,将内容构建成html,再转... 目录一、为什么要搞?二、准备如何搞?三、说搞咱就搞!抓取文章提取内容构建html转存markdown

Python获取中国节假日数据记录入JSON文件

《Python获取中国节假日数据记录入JSON文件》项目系统内置的日历应用为了提升用户体验,特别设置了在调休日期显示“休”的UI图标功能,那么问题是这些调休数据从哪里来呢?我尝试一种更为智能的方法:P... 目录节假日数据获取存入jsON文件节假日数据读取封装完整代码项目系统内置的日历应用为了提升用户体验,

Python FastAPI+Celery+RabbitMQ实现分布式图片水印处理系统

《PythonFastAPI+Celery+RabbitMQ实现分布式图片水印处理系统》这篇文章主要为大家详细介绍了PythonFastAPI如何结合Celery以及RabbitMQ实现简单的分布式... 实现思路FastAPI 服务器Celery 任务队列RabbitMQ 作为消息代理定时任务处理完整

Python Websockets库的使用指南

《PythonWebsockets库的使用指南》pythonwebsockets库是一个用于创建WebSocket服务器和客户端的Python库,它提供了一种简单的方式来实现实时通信,支持异步和同步... 目录一、WebSocket 简介二、python 的 websockets 库安装三、完整代码示例1.

揭秘Python Socket网络编程的7种硬核用法

《揭秘PythonSocket网络编程的7种硬核用法》Socket不仅能做聊天室,还能干一大堆硬核操作,这篇文章就带大家看看Python网络编程的7种超实用玩法,感兴趣的小伙伴可以跟随小编一起... 目录1.端口扫描器:探测开放端口2.简易 HTTP 服务器:10 秒搭个网页3.局域网游戏:多人联机对战4.

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

Python使用自带的base64库进行base64编码和解码

《Python使用自带的base64库进行base64编码和解码》在Python中,处理数据的编码和解码是数据传输和存储中非常普遍的需求,其中,Base64是一种常用的编码方案,本文我将详细介绍如何使... 目录引言使用python的base64库进行编码和解码编码函数解码函数Base64编码的应用场景注意

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.