之前,应朋友所托,完成个四边形面积计算程序,于是不由自主考虑来个扩展,解决任意多边形面积的计算。
一开始想到了某定点的三角形剖分,但遇到凹凸多边形引发的多种情况,过于复杂,放弃。
后来想到用图形学中填充算法中的扫描线方法,切分成梯形与三角形,将交点存入活性边表后再计算面积,感觉也较复杂,放弃。
再然后,找到个计算几何大神O’Rourke在1998年公开的成果。
*(书名:Computational Geometry in C,第二版P20)*
1-原理介绍
上书中给出定理:
任意多边形的面积可由任意一点与多边形上依次两点连线构成的三角形矢量面积求和得出。
矢量面积=三角形两边矢量的叉乘。
如下图:
按定理,多边形面积由P点与A-G的各顶点连接所构成的三角形矢量面积构成,假定多边形顶点坐标顺序为A-G,逆时针为正方向,则有如下结论:
PAB,PBC,PCD均为顺时针,面积为负;
PDE,PEF,PFG,PGA均未逆时针,面积为正;
但无论正负,均可通过P点与顶点连线的矢量叉乘完成,叉乘结果中已包含面积的正负。
2-程序设计
采用C++的vector(动态数组)存储顶点坐标。
为方便计算,直接将P点定为原点(0,0),则多边形顶点xy坐标即为向量在xy上分量。
循环计算多边形顶点坐标每一点与下一点之间的线段,及这两点与P连线的矢量所围成的三角形面积。
计算面积的函数代码如下:
iArea=iArea+(vecPoly[iCycle].x*vecPoly[(iCycle+1) % iCount].y-vecPoly[(iCycle+1) % iCount].x*vecPoly[iCycle].y); int intAreaCalc(vector<myPoint> &vecPoly) { int iCycle,iCount,iArea; iCycle=0; iArea=0; iCount=vecPoly.size(); for(iCycle=0;iCycle<iCount;iCycle++) { iArea=iArea+(vecPoly[iCycle].x*vecPoly[(iCycle+1) % iCount].y-vecPoly[(iCycle+1) % iCount].x*vecPoly[iCycle].y); } return abs(0.5*iArea); }
注意,要注意的是最后一个顶点,要与第一个顶点练成三角形,可将循环变量对顶点总数求同余,则循环过程中的最后一点+1后,自然会成为第一个顶点,上述代码中的“% iCount”即为解决此问题。
完整程序,请下载工程文件。
http://files.cnblogs.com/vbspine/sdkPoly.rar
Ps:上述程序在Win7x64,VS2008环境下编译通过。