圆的面积并
By momodi at 2009.11
摘要
设想生活中往往需要知道很多图形的面积,但是当图形有多个的情况下,我们怎么知道这些图形的面积的并集呢? 现实中的图形往往可以转化成一些多边形和圆的组合. 本文介绍了一个O(n^2logn)求圆的面积并的算法. 并将其拓展,使其可以求得复杂现实图形的并集.与经典的扫描线相比,本文介绍的方法效率高,占用空间少,容易实现及拓展,而且几乎没有难以处理的几何特殊情况.
有向面积和简单多边形的面积
凸多边形的面积可以通过三角剖分(如下图所示)得到的每一个三角形的面积相加来得到.
上图中是以多边形内部的一个点为中心来进行的三角剖分,剖分出来的所有三角形后,只要把面积简单相加就可以了. 利用有向面积,我们可以把这个点取在多边形外部.
我们可以用如下公式来表示凸多边形的面积:
)
½(XkYk+1 – Xk+1Yk)便是一个以原点为中心点进行剖分出来的三角形的有向面积. 其实,这便是两个向量的叉积的值一半.
如果求得的总面积为正, 那么多边形的点集为逆时针存储. 否则为顺时针.
这个公式同样适用于简单多边形. 并不要求多边形一定是凸的.
圆的面积并的算法
先来观察一个图形:
此图表示了三个圆形的并. 我们在圆上随机选取了一些点, 并且使其包含圆与圆之间的交点. 我们选取了最外层的点对, 并将其用线段连接起来.
从图中我们可以看出, 我们将这个图形剖分成了一个简单多边形和一些半月. 对于这个图形, 我们只要求出简单多边形的面积再加上所有的半月的面积. 便是这三个圆形的面积并.
我们将这个做法一般话.
1. 去掉重复的圆形并求出圆与圆之间的所有交点的点集Ω.
2. 对于任意一个圆, 点集Ω中对应于这个圆的交点将其离散化成一个个小圆弧. 如果圆弧的角度大于90度, 则添加虚点, 将其分解成两个小圆弧.
3. 规定逆时针为圆弧的正方向.对于所有的小圆弧, 判断其是不是被其他圆所覆盖过. 取出所有的未被覆盖过的圆弧.
4. 对于圆弧的两个端点连接而成的线段, 将其看成简单多边形的一条边, 把其构成的有向面积加到总面积里面.
5. 对于圆弧所构成的半月, 将其面积直接加到总面积里面.
判断小圆弧是不是被其他的圆覆盖的方法, 可以取出这个圆弧的特征点(比如圆弧的中点), 然后判断这个点是不是被其他圆所覆盖.因为小圆弧是用所有交点离散化之后的,所以小圆弧上除了端点之外都有共同特征,因而上述方法是正确的.
这个做法是否适用于所有的情况呢? 且看如下图形:
对于这种内空的情况, 我们的做法也是适用的.因为对于内部的图形来说. 我们求得的多边形是顺时针的. 所以最后的总面积相当于减掉了这部分面积. 下面让我们来证明这个做法的正确性.
算法复杂度
对于每一个圆来说, 其圆上的点最多是O(n)个. 也就是说圆上那个最多有O(n)条小圆弧.每一个圆弧判断是否被覆盖的复杂度是O(n). 也就是说总的复杂度是O(n^3), n为圆的个数.
实际上,这样裸露的实现方法浪费了大量的有用信息,所以算法复杂度过高.我们可以通过改进圆弧是否被覆盖的方法来优化时间复杂度.
算法的优化
对于一个圆形来说,把他和所有其他圆形的交点求出来之后我们可以知道这个圆形上所有被覆盖掉的极角区间.我们可以用O(nlogn)的算法来求出极角区间的并.再用总的区间减掉这个并集,剩下的就是我们所要知道的所有未被覆盖过的小圆弧.
经过这样的优化,时间复杂度变为了O(n^2logn).而且这个logn的出现是因为区间并需要排序.如果用非基于比较的排序算法的话,时间复杂度可以降低为O(n^2).
算法的拓展
相信读者已经能够看得出, 此算法有一定的通用性. 如果把圆换成多边形, 最终算法的框架也是一样的. 只是细节上会有所变化.甚至可以是求圆和多边形等混合图形的并集.
Sun, 22 May 2022 04:45:14 +0800
I do trust all of the ideas you've introduced for your post. They're very convincing and will definitely work. Still, the posts are very short for beginners. May you please prolong them a bit from next time? Thank you for the post. How to Be a Payment Processor
Sat, 28 May 2022 05:41:26 +0800
I like this site so much,i bookmarked on this website. 먹튀검증
Sun, 05 Jun 2022 05:26:56 +0800
Thanks for the post, was an interesting read. Curious as to how you came about that solution… 麥克風
Tue, 14 Jun 2022 17:33:34 +0800
I do believe all of the ideas you’ve offered to your post. They’re really convincing and will certainly work. Still, the posts are very brief for starters. May just you please lengthen them a little from next time? Thank you for the post. 回收手提電腦
Fri, 24 Jun 2022 00:44:24 +0800
very nice put up, i actually love this web site, keep on it day spa in fort lauderdale florida
Fri, 29 Jul 2022 15:39:26 +0800
I got what you intend, thanks for putting up. Woh I am glad to find this website through google. https://voyance-tel-avenir.com
Sat, 20 Aug 2022 00:08:09 +0800
I am usually to blogging and i also truly appreciate your articles. This great article has really peaks my interest. My goal is to bookmark your blog and maintain checking for brand new details. 빅토리카지노먹튀
Fri, 26 Aug 2022 22:08:47 +0800
This internet site is my breathing in, really good layout and perfect content . 온라인카지노
Mon, 03 Oct 2022 23:31:43 +0800
After study a few of the content for your website now, i genuinely as if your technique of blogging. I bookmarked it to my bookmark website list and are checking back soon. Pls have a look at my website too and told me if you agree. Laptop