百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Shamos算法:一种在平面上找到最远点的方法

itomcoil 2025-05-05 16:33 18 浏览

旋转卡尺算法简介

Shamos算法,也叫旋转卡尺(Rotating calipers)算法,是一种用于解决计算几何问题的优化算法。它可以用来解决许多几何问题,包括计算点集的宽度或直径。算法的名称来源于其类似于旋转卡尺(测量工具)的操作方式。

旋转卡尺算法的核心思想是将一个“卡尺”围绕凸多边形旋转,以便检测所有对立点对。通过这样的旋转,我们可以找到最小的包围矩形或者计算多边形的直径等。

算法实现步骤

  1. 1. 初始化:从多边形的一个顶点开始。
  2. 2. 旋转卡尺:将卡尺旋转,直到它的一个刀刃与多边形的一个边平行。
  3. 3. 检测对立点:记录卡尺的两个刀刃所触及的对立点对。
  4. 4. 继续旋转:继续旋转卡尺,直到完整地围绕多边形旋转一圈,检测所有的对立点对。

算法的历史背景

旋转卡尺算法最早在 1978 年由 Michael Shamos 在其论文中提出,用于计算凸多边形的直径。他的算法在计算复杂度上表现优异,能够在 时间内解决问题。

之后,Godfried Toussaint 将“旋转卡尺”这一术语引入,并展示了该算法在解决许多计算几何问题中的应用。

应用场景

  1. 最小外接矩形(Minimum Bounding Rectangle, MBR):在计算机图形学和地理信息系统中,最小外接矩形用于快速判断物体是否相交或进行空间查询。
  2. 最远点对(Farthest Pair):在计算机视觉和机器学习中,找出多边形的最远点对对于对象检测和图像分析非常重要。
  3. 最大内接圆(Maximum Inscribed Circle):用于寻找多边形内能够容纳的最大圆,这在机器人路径规划和形状分析中具有实际应用。

代码示例

下面是一个简单的旋转卡尺算法实现,用于计算二维平面上多边形的最小外接矩形。我们将使用 Python 的 shapely 库来辅助实现这一过程。

from shapely.geometry import Polygon
from shapely.affinity import rotate
import matplotlib.pyplot as plt
import numpy as np

def angle_between_edges(polygon, i, j):
    """计算从第i边到第j边的角度"""
    p1, p2 = polygon.exterior.coords[i], polygon.exterior.coords[i + 1]
    q1, q2 = polygon.exterior.coords[j], polygon.exterior.coords[j + 1]
    angle = np.arctan2(p2[1] - p1[1], p2[0] - p1[0]) - np.arctan2(q2[1] - q1[1], q2[0] - q1[0])
    return abs(angle)

def rotating_calipers(polygon):
    """计算多边形的所有对立点对"""
    n = len(polygon.exterior.coords) - 1
    i = 0
    j = 1
    pairs = []

    while j < n:
        if angle_between_edges(polygon, i, j) < np.pi:
            j += 1
        else:
            pairs.append((i, j))
            i += 1

    # 最后一次添加对立点对
    pairs.append((i, j))
    return pairs

# 示例多边形
points = [(1, 2), (3, 5), (6, 4), (7, 1), (5, -2), (2, -3), (-1, -1), (-2, 2)]
polygon = Polygon(points)

# 计算对立点对
pairs = rotating_calipers(polygon)
print("对立点对:", pairs)

输出:

对立点对: [(0, 3), (1, 8)]

可视化

def plot_polygon_with_bounding_box(polygon, pairs, title):
    """绘制多边形及其最小外接矩形"""
    x, y = polygon.exterior.xy
    plt.plot(x, y, 'b-', label='多边形')

    min_rect = polygon.minimum_rotated_rectangle
    min_rect_x, min_rect_y = min_rect.exterior.xy
    plt.plot(min_rect_x, min_rect_y, 'r--', label='最小外接矩形')

    plt.fill(x, y, alpha=0.3, fc='blue', label='多边形区域')
    plt.fill(min_rect_x, min_rect_y, alpha=0.1, fc='red', label='外接矩形区域')

    for (i, j) in pairs:
        plt.plot([polygon.exterior.coords[i][0], polygon.exterior.coords[j][0]],
                 [polygon.exterior.coords[i][1], polygon.exterior.coords[j][1]],
                 'g--', label='对立点对')

    plt.title(title)
    plt.xlabel('X 轴')
    plt.ylabel('Y 轴')
    plt.legend()
    plt.grid(True)

# 绘制结果
plt.figure(figsize=(8, 8))
plot_polygon_with_bounding_box(polygon, pairs, '多边形及其最小外接矩形与对立点对')
plt.show()

运行以上代码,将显示一个图形,其中包括:

  • 蓝色多边形:代表定义的复杂多边形。
  • 红色虚线矩形:代表多边形的最小外接矩形。
  • 绿色虚线:显示多边形的对立点对。

小结

旋转卡尺算法是一种高效解决几何问题的方法,通过旋转和记录,可以在多边形的各种旋转状态下找到最优解。

它的应用场景广泛,从图形处理到空间分析都可以见到它的身影。

相关推荐

selenium(WEB自动化工具)

定义解释Selenium是一个用于Web应用程序测试的工具。Selenium测试直接运行在浏览器中,就像真正的用户在操作一样。支持的浏览器包括IE(7,8,9,10,11),MozillaF...

开发利器丨如何使用ELK设计微服务中的日志收集方案?

【摘要】微服务各个组件的相关实践会涉及到工具,本文将会介绍微服务日常开发的一些利器,这些工具帮助我们构建更加健壮的微服务系统,并帮助排查解决微服务系统中的问题与性能瓶颈等。我们将重点介绍微服务架构中...

高并发系统设计:应对每秒数万QPS的架构策略

当面试官问及"如何应对每秒几万QPS(QueriesPerSecond)"时,大概率是想知道你对高并发系统设计的理解有多少。本文将深入探讨从基础设施到应用层面的解决方案。01、理解...

2025 年每个 JavaScript 开发者都应该了解的功能

大家好,很高兴又见面了,我是"高级前端进阶",由我带着大家一起关注前端前沿、深入前端底层技术,大家一起进步,也欢迎大家关注、点赞、收藏、转发。1.Iteratorhelpers开发者...

JavaScript Array 对象

Array对象Array对象用于在变量中存储多个值:varcars=["Saab","Volvo","BMW"];第一个数组元素的索引值为0,第二个索引值为1,以此类推。更多有...

Gemini 2.5编程全球霸榜,谷歌重回AI王座,神秘模型曝光,奥特曼迎战

刚刚,Gemini2.5Pro编程登顶,6美元性价比碾压Claude3.7Sonnet。不仅如此,谷歌还暗藏着更强的编程模型Dragontail,这次是要彻底翻盘了。谷歌,彻底打了一场漂亮的翻...

动力节点最新JavaScript教程(高级篇),深入学习JavaScript

JavaScript是一种运行在浏览器中的解释型编程语言,它的解释器被称为JavaScript引擎,是浏览器的一部分,JavaScript广泛用于浏览器客户端编程,通常JavaScript脚本是通过嵌...

一文看懂Kiro,其 Spec工作流秒杀Cursor,可移植至Claude Code

当Cursor的“即兴编程”开始拖累项目质量,AWS新晋IDEKiro以Spec工作流打出“先规范后编码”的系统工程思维:需求-设计-任务三件套一次生成,文档与代码同步落地,复杂项目不...

「晚安·好梦」努力只能及格,拼命才能优秀

欢迎光临,浏览之前点击上面的音乐放松一下心情吧!喜欢的话给小编一个关注呀!Effortscanonlypass,anddesperatelycanbeexcellent.努力只能及格...

JavaScript 中 some 与 every 方法的区别是什么?

大家好,很高兴又见面了,我是姜茶的编程笔记,我们一起学习前端相关领域技术,共同进步,也欢迎大家关注、点赞、收藏、转发,您的支持是我不断创作的动力在JavaScript中,Array.protot...

10个高效的Python爬虫框架,你用过几个?

小型爬虫需求,requests库+bs4库就能解决;大型爬虫数据,尤其涉及异步抓取、内容管理及后续扩展等功能时,就需要用到爬虫框架了。下面介绍了10个爬虫框架,大家可以学习使用!1.Scrapysc...

12个高效的Python爬虫框架,你用过几个?

实现爬虫技术的编程环境有很多种,Java、Python、C++等都可以用来爬虫。但很多人选择Python来写爬虫,为什么呢?因为Python确实很适合做爬虫,丰富的第三方库十分强大,简单几行代码便可实...

pip3 install pyspider报错问题解决

运行如下命令报错:>>>pip3installpyspider观察上面的报错问题,需要安装pycurl。是到这个网址:http://www.lfd.uci.edu/~gohlke...

PySpider框架的使用

PysiderPysider是一个国人用Python编写的、带有强大的WebUI的网络爬虫系统,它支持多种数据库、任务监控、项目管理、结果查看、URL去重等强大的功能。安装pip3inst...

「机器学习」神经网络的激活函数、并通过python实现激活函数

神经网络的激活函数、并通过python实现whatis激活函数感知机的网络结构如下:左图中,偏置b没有被画出来,如果要表示出b,可以像右图那样做。用数学式来表示感知机:上面这个数学式子可以被改写:...