凸包问题 Graham Scan
2020-01-09 15:14:21
凸包问题是计算几何的核心问题,并且凸包问题的研究已经持续了好多年,这中间涌现出了一大批优秀的算法。
凸包问题的最优解法是Graham Scan算法,该算法可以保证在最差情况下也能在O(nlogn)的时间复杂度求出结果。
Graham Scan算法的核心思路有两个步骤:
1. 预处理:将所有的点按照某个给定的点根据夹角进行排序;
2. 在排序好的结果上,按照顺序逆时针依次进行判断(to left test),将符合条件的节点加入栈中;
在具体实现的时候有一个非常好用的trick,就是不直接对夹角进行排序,而是假定在 +inf 和 -inf 有两个点,那么我们只需要根据x轴进行排序,就可以得到它们相对于两个无穷远点的夹角排序。
这样我们就可以直接使用第二步的递推迭代得到upper hull和lower hull,最后将这两个结果拼接起来就是最终的结果。
下面根据一条leetcode的题目来具体看下代码实现。
问题描述:


问题求解:
public int[][] outerTrees(int[][] points) {
if (points.length <= 3) return points;
int n = points.length;
Arrays.sort(points, new Comparator<int[]>(){
public int compare(int[] o1, int[] o2) {
return o1[0] == o2[0] ? o1[1] - o2[1] : o1[0] - o2[0];
}
});
Stack<int[]> upper = new Stack<>();
Stack<int[]> lower = new Stack<>();
Stack<int[]> left = new Stack<>();
upper.add(points[n - 1]);
upper.add(points[n - 2]);
lower.add(points[0]);
lower.add(points[1]);
for (int i = 0; i < n - 2; i++) left.add(points[i]);
while (!left.isEmpty()) {
int[] curr = left.pop();
if (upper.size() == 1) {
upper.add(curr);
continue;
}
int[] p2 = upper.pop();
int[] p1 = upper.pop();
if (to_left_test(p1, p2, curr)) {
upper.add(p1);
upper.add(p2);
upper.add(curr);
}
else {
upper.add(p1);
left.add(curr);
}
}
for (int i = n - 1; i > 1; i--) left.add(points[i]);
while (!left.isEmpty()) {
int[] curr = left.pop();
if (lower.size() == 1) {
lower.add(curr);
continue;
}
int[] p2 = lower.pop();
int[] p1 = lower.pop();
if (to_left_test(p1, p2, curr)) {
lower.add(p1);
lower.add(p2);
lower.add(curr);
}
else {
lower.add(p1);
left.add(curr);
}
}
Set<int[]> set = new HashSet<>();
set.addAll(upper);
set.addAll(lower);
int[][] res = new int[set.size()][2];
int idx = 0;
for (int[] p : set) {
res[idx][0] = p[0];
res[idx][1] = p[1];
idx += 1;
}
return res;
}
private boolean to_left_test(int[] p1, int[] p2, int[] p3) {
return p1[0] * p2[1] + p1[1] * p3[0] + p2[0] * p3[1] -
p2[1] * p3[0] - p1[1] * p2[0] - p1[0] * p3[1] >= 0;
}
凸包问题 Graham Scan的更多相关文章
- 凸包问题——Graham Scan
Graham Scan 概述: 对于凸多边形的定义不在这里做详细叙述,这里给出算法的实现原理. Step 1: 找出x值最小的点的集合,从其中找出y值最小的点作为初始点 Step 2: 获得新序列后, ...
- Graham Scan凸包算法
获得凸包的算法可以算是计算几何中最基础的算法之一了.寻找凸包的算法有很多种,Graham Scan算法是一种十分简单高效的二维凸包算法,能够在O(nlogn)的时间内找到凸包. 首先介绍一下二维向量的 ...
- 凸包Graham Scan算法实现
凸包算法实现点集合中搜索凸包顶点的功能,可以处理共线情况,可以输出共线点也可以不输出而只输出凸包顶点.经典的Graham Scan算法,点排序使用极角排序方式,并对共线情况做特殊处理.一般算法是将共线 ...
- 凸包模板——Graham扫描法
凸包模板--Graham扫描法 First 标签: 数学方法--计算几何 题目:洛谷P2742[模板]二维凸包/[USACO5.1]圈奶牛Fencing the Cows yyb的讲解:https:/ ...
- 计算几何--求凸包模板--Graham算法--poj 1113
Wall Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 28157 Accepted: 9401 Description ...
- 凸包入门(Graham扫描法)(A - Wall POJ - 1113)
题目链接:https://cn.vjudge.net/contest/276359#problem/A 题目大意:有一个国王,要在自己的城堡周围建立围墙,要求围墙能把城堡全部围起来,并且围墙距离城堡的 ...
- poj1113Wall 求凸包周长 Graham扫描法
#include<iostream> #include<algorithm> #include<cmath> using namespace std; typede ...
- HDU 1348 Wall ( 凸包周长 )
链接:传送门 题意:给出二维坐标轴上 n 个点,这 n 个点构成了一个城堡,国王想建一堵墙,城墙与城堡之间的距离总不小于一个数 L ,求城墙的最小长度,答案四舍五入 思路:城墙与城堡直线长度是相等的, ...
- 计算几何(一):凸包问题(Convex Hull)
引言 首先介绍下什么是凸包?如下图: 在一个二维坐标系中,有若干点杂乱排列着,将最外层的点连接起来构成的凸多边型,它能包含给定的所有的点,这个多边形就是凸包. 实际上可以理解为用一个橡皮筋包含住所有给 ...
随机推荐
- nginx增加访问验证
使用OpenSSL实用程序创建密码文件 如果您的服务器上安装了OpenSSL,则可以创建没有附加软件包的密码文件.我们将在/ etc / nginx配置目录中创建一个名为.htpasswd的隐藏文件来 ...
- JavaScript 設計模型 - Iterator
Iterator Pattern是一個很重要也很簡單的Pattern:迭代器!我們可以提供一個統一入口的迭代器,Client只需要知道有哪些方法,或是有哪些Concrete Iterator,並不需要 ...
- Go的sync
关于 pool 的由来可以参考: github issues 文章 sync.Pool 的作用及为什么要用到它 Rob Pike 扩展了sync.pool 类型的文档,并且将其目的描述得更清楚: Po ...
- RHEL系统下安装atlassian-jira-5
操作系统:RHEL 6.4 x86_64 Jira版本:atlassian-jira-5.2.11-x64.bin 安装路径:/opt/atlassian/jira/ 数据保存路径:/opt/atla ...
- Spring Boot 2.x基础教程:使用 Thymeleaf开发Web页面
通过本系列教程的前几章内容(API开发.数据访问).我们已经具备完成一个涵盖数据存储.提供HTTP接口的完整后端服务了.依托这些技能,我们已经可以配合前端开发人员,一起来完成一些前后端分离的Web项目 ...
- Flutter Widgets 之 ListWheelScrollView
注意:无特殊说明,Flutter版本及Dart版本如下: Flutter版本: 1.12.13+hotfix.5 Dart版本: 2.7.0 基础用法 在展示大量数据的时候我们第一会想到使用ListV ...
- 这几个IDEA高级调试技巧,用完就是香
一个项目启动两次 测试分布式项目时,经常要一个项目启动2次,不用将一个项目打开多次启动,配置一下即可 1.点击Edit Configurations 2.勾选Allow parallel run 3. ...
- 机器学习基础——详解自然语言处理之tf-idf
本文始发于个人公众号:TechFlow,原创不易,求个关注 今天的文章和大家聊聊文本分析当中的一个简单但又大名鼎鼎的算法--TF-idf.说起来这个算法是自然语言处理领域的重要算法,但是因为它太有名了 ...
- 树莓派3B安装OpenWrt打造超级路由器
网上有很多树莓派安装OpenWrt的教程,我这里写一下个人安装体验以及踩过的坑
- [Python之路] object类中的特殊方法
一.object类的源码 python版本:3.8 class object: """ The most base type """ # d ...