判断点在多边形内的算法(Winding Number详解)

2024-09-03 03:58

本文主要是介绍判断点在多边形内的算法(Winding Number详解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

在计算几何中,判定点是否在多边形内,是个非常有趣的问题。通常有两种方法:

1.Crossing Number(交叉数)

它计算从点P开始的射线穿过多边形边界的次数。当“交叉数”是偶数时,点在外面;当它是奇数时,点在里面。这种方法有时被称为“奇-偶”检验。

2.Winding Number(环绕数)

它计算多边形绕着点P旋转的次数。只有当“圈数”wn = 0时,点才在外面; 否则,点在里面。

如果一个多边形是不自交的(称为“简单多边形”),那么这两种方法对任意点都给出相同的结果。但对于非简单多边形,这两种方法在某些情况下会给出不同的答案。如下图所示,当一个多边形与自身重叠时,对于重叠区域内的点,如果使用交叉数判断,它在外面;而使用环绕数判断则在里面。

 

         

                                     顶点按次序编号: 0 1 2 3 4 5 6 7 8 9

在上图中,绿色区域中的点,wn = 2,表示在多边形中重叠了2次。相比于Crossing number,winding number给出了更内蕴性的答案。

尽管如此,早些时候,crossing number方法应用的更广泛,因为最初计算几何专家们错误地认为crossing number比winding number计算起来更加高效。但事实并非如此,两者的时间复杂度完全一样。Franklin在2000年给出一个计算winding number的非常快的实现。因此,为了几何正确性和效率的原因,在确定一个多边形中的一个点时,wn算法应该总是首选的。

The Crossing Number

该方法计算从点P开始的射线穿过多边形边界的次数(不管穿过的方向)。如果这个数是偶数,那么点在外面;否则,当交叉数为奇数时,点在多边形内。其正确性很容易理解,因为每次射线穿过多边形边缘时,它的内外奇偶性都会发生变化(因为边界总是分隔内外)。最终,任何射线都在边界多边形之外结束。所以,如果点在多边形内,那么对边界的穿过次序一定是:out>...>in>out,因此交叉数一定是奇数;同样地,如果点在多边形外,那么对边界的穿过次序一定是in > out ... > in > out,因此交叉数必是偶数。

在实现crossing number的算法时,必须确保只计算改变奇偶性的交叉位置。特别是,对于射线穿过顶点的情况需要适当的处理。下图列举了射线与多边形可能的相交情况:

                                                                        射线与多边形的边可能的相交情况  

此外,必须确定多边形边界上的点P是在内部还是外部。一般约定:如果点在边的左侧,那么认为点P在内部;如果点在边的右侧,那么认为点P在外部。如果两个不同的多边形共享一个共同的边界线段,那么该线段上的一点将会在一个多边形或另一个多边形中,而不是同时在两个多边形中。这避免了许多可能发生的问题,特别是在计算机图形显示中。

一个简单的做法是选择一条x轴正方向的水平射线,对于这样一条射线,很容易计算多边形的边与它的交点。而且,很容易确定交点是否存在。算法只需沿着多边形的每一条边,依次计算交点,当相交时,cn增加1,从而计算出最终的总交叉数。

此外,相交测试必须遵循如下的规则,处理一些特殊情况(如上图):

  1. 向上的边,包含起点,但不包含终点;
  2. 向下的边,包含终点,但不包含起点;
  3. 水平的边,不包含起点和终点;
  4. 边与射线的交点必须严格在点P的右侧

按照上述规则,处理特殊的相交情况,就能得到正确的交叉数。其中,规则#4将导致在边界右侧的点在多边形外部,在左侧的点将会被判定为在内部。

Crossing Number Pseudo-Code:

对于n个点组成的多边形V={V[0], V[1], ......,V[n]},其中V[n]=V[0], 计算几何大牛Franklin给出了一个非常有名的实现:

typedef struct {int x, y;} Point;cn_PnPoly( Point P, Point V[], int n )
{int    cn = 0;    // the  crossing number counter// loop through all edges of the polygonfor (each edge E[i]:V[i]V[i+1] of the polygon) {if (E[i] crosses upward ala Rule #1|| E[i] crosses downward ala  Rule #2) {if (P.x <  x_intersect of E[i] with y=P.y)   // Rule #4++cn;   // a valid crossing to the right of P.x}}return (cn&1);    // 0 if even (out), and 1 if  odd (in)}

注意,对于满足规则#1和#2的向上和向下交叉的测试也排除了水平边缘(规则#3)。总而言之,很多工作是通过几个测试完成的,这使得这个算法很优雅。

然而,交叉数方法的有效性是基于“约当曲线定理”(Jordan Curve Theorem),该定理表明,一条简单的闭合曲线将二维平面分成两个完全连通的分量:一个有界的“内”分量和一个无界的“外”分量。需要注意的是,曲线必须是简单的(没有自身交叉),否则可能有两个以上的组成部分,然后就不能保证跨越边界改变进出奇偶性。因此,该方法不适用于自相交的多边形。

The Winding Number

另一方面,winding number方法能准确判定一个点是否在自交的封闭曲线内。该方法通过计算多边形有多少次环绕点P来实现。只有当多边形不环绕该点,也就是环绕数wn = 0时,一个点才在外面。

不妨定义:平面上的点P相对于任意连续封闭曲线的环绕数为\mathbf{wn}(P,C)。对于一条水平向右的射线R,我们每一条与R相交的边需要判断其终点在R上面还是下面。如果边从下往上穿过R,wn+1;否则wn-1。所有边遍历一遍,最终得到总的\mathbf{wn}(P,C),如下图所示:

此外,我们没必要计算实际的交点,只需要使用如下方法判断当前穿过的边的环绕数应该+1还是-1:

如下图所示,如果一条边向上穿过射线R,那么P点在边ViVi+1的左侧;而对于一条向下的边,P点在边ViVi+1的右侧。

 

Winding Number Pseudo-Code:

通过以上分析,容易给出如下的wn计算伪代码(和cn的计算一样使用相同的边相交规则):


typedef struct {int x, y;} Point;wn_PnPoly( Point P, Point V[], int n )
{int    wn = 0;    // the  winding number counter// loop through all edges of the polygonfor (each edge E[i]:V[i]V[i+1] of the polygon) {if (E[i] crosses upward ala Rule #1)  {if (P is  strictly left of E[i])    // Rule #4++wn;   // a valid up intersect right of P.x}elseif (E[i] crosses downward ala Rule  #2) {if (P is  strictly right of E[i])   // Rule #4--wn;   // a valid down intersect right of P.x}}return wn;    // =0 <=> P is outside the polygon}

显然,环绕数方法与交叉数方法有着相同的计算效率。但由于该方法更加具有普遍性,因此,在确定一个点是否在任意多边形内时,推荐使用Winding Number方法。

通过一些技巧可以进一步提高wn算法的效率,在下面给出的wn_PnPoly() 的实现中,我们可以看到这一点。在该代码中,所有完全在P以上或完全在P以下的边只经过两次不等式检验就被拒绝(没有交点)。然而,在目前流行的cn算法的实现中,需要3次不等式检验才能做到这一点。由于在实际应用中,大多数边都会被拒绝,因此进行比较的次数减少了大约33%(或更多)。在使用非常大的(1,000,000边)随机多边形(边长<多边形直径的1/10)和1000个随机测试点(在多边形的边界内)进行运行时测试时,测试结果表明wn算法的平均效率提高了20%。

 

Winding Number算法的实现

// isLeft(): tests if a point is Left|On|Right of an infinite line.
//    Input:  three points P0, P1, and P2
//    Return: >0 for P2 left of the line through P0 and P1
//            =0 for P2  on the line
//            <0 for P2  right of the lineinline int
isLeft( Point P0, Point P1, Point P2 )
{return ( (P1.x - P0.x) * (P2.y - P0.y)- (P2.x -  P0.x) * (P1.y - P0.y) );
}
//===================================================================// cn_PnPoly(): crossing number test for a point in a polygon
//      Input:   P = a point,
//               V[] = vertex points of a polygon V[n+1] with V[n]=V[0]
//      Return:  0 = outside, 1 = inside
// This code is patterned after [Franklin, 2000]
int
cn_PnPoly( Point P, Point* V, int n )
{int    cn = 0;    // the  crossing number counter// loop through all edges of the polygonfor (int i=0; i<n; i++) {    // edge from V[i]  to V[i+1]if (((V[i].y <= P.y) && (V[i+1].y > P.y))     // an upward crossing|| ((V[i].y > P.y) && (V[i+1].y <=  P.y))) { // a downward crossing// compute  the actual edge-ray intersect x-coordinatefloat vt = (float)(P.y  - V[i].y) / (V[i+1].y - V[i].y);if (P.x <  V[i].x + vt * (V[i+1].x - V[i].x)) // P.x < intersect++cn;   // a valid crossing of y=P.y right of P.x}}return (cn&1);    // 0 if even (out), and 1 if  odd (in)}
//===================================================================// wn_PnPoly(): winding number test for a point in a polygon
//      Input:   P = a point,
//               V[] = vertex points of a polygon V[n+1] with V[n]=V[0]
//      Return:  wn = the winding number (=0 only when P is outside)
int
wn_PnPoly( Point P, Point* V, int n )
{int    wn = 0;    // the  winding number counter// loop through all edges of the polygonfor (int i=0; i<n; i++) {   // edge from V[i] to  V[i+1]if (V[i].y <= P.y) {          // start y <= P.yif (V[i+1].y  > P.y)      // an upward crossingif (isLeft( V[i], V[i+1], P) > 0)  // P left of  edge++wn;            // have  a valid up intersect}else {                        // start y > P.y (no test needed)if (V[i+1].y  <= P.y)     // a downward crossingif (isLeft( V[i], V[i+1], P) < 0)  // P right of  edge--wn;            // have  a valid down intersect}}return wn;
}
//===================================================================

 

References

Wm. Randolph Franklin, "PNPOLY  - Point Inclusion in Polygon Test" Web Page (2000)

Tomas Moller & Eric Haines, "Ray/Polygon Intersection" in Real-Time Rendering (3rd Edition) (2008)

Joseph O'Rourke, "Point in  Polygon" in Computational Geometry in C (2nd Edition) (1998)

这篇关于判断点在多边形内的算法(Winding Number详解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig

usaco 1.2 Name That Number(数字字母转化)

巧妙的利用code[b[0]-'A'] 将字符ABC...Z转换为数字 需要注意的是重新开一个数组 c [ ] 存储字符串 应人为的在末尾附上 ‘ \ 0 ’ 详见代码: /*ID: who jayLANG: C++TASK: namenum*/#include<stdio.h>#include<string.h>int main(){FILE *fin = fopen (

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

poj 3259 uva 558 Wormholes(bellman最短路负权回路判断)

poj 3259: 题意:John的农场里n块地,m条路连接两块地,w个虫洞,虫洞是一条单向路,不但会把你传送到目的地,而且时间会倒退Ts。 任务是求你会不会在从某块地出发后又回来,看到了离开之前的自己。 判断树中是否存在负权回路就ok了。 bellman代码: #include<stdio.h>const int MaxN = 501;//农场数const int