问题:给出若干个矩形,(给的是矩形左上角和右下角坐标),求最后所得图形的面积/周长;
三个矩形如左图所示,而若要计算面积,看右图,用3个矩形各自的面积之和减去重复部分(红色和蓝色)的面积
人算很简单,但是用算法怎么实现呢?
此类问题一般都是用线段树辅助扫描法来计算;
什么是扫描法?有什么用?怎么用?
可以想象成一根假想的线,将图从左往右或从右往左或自下而上或自上而下“扫描”一遍,至于扫描的是什么则根据具体应用选择。
扫描线可以计算矩形面积、周长,可以计算线段交点,可以实现多边形扫描转换,在图的处理方面经常用到。
这里总结一下扫描线计算矩形面积和周长的算法。
怎么用?首先,对于之前的图,除了用总面积减去重合面积,还可以换一种计算方法,如图:
此图用4条横线将整个图划分成了5个部分,显然此时再算面积就可以用各个颜色的部分求和。
想想,这样计算的整个慢过程:
假设我们的视线自下而上,首先,我们看到了最下面灰色矩形的下边,
用这个下边的长度乘以这条边和上一条边的高度差即得到灰色矩形面积,
继续看到蓝色的矩形的下边,虽然蓝色矩形有两个,但我们计算时自然会用结合律将两个矩形的下边加起来再去乘以同样的高,
然后重复这样的操作,我们最终可以求得整个图形的面积。
但是,这依旧是人做的,计算机要怎么实现呢?
首先的问题是,计算机要怎么保存这张图这些矩形?
从刚才的过程,我们不难发现,我们只需要保存这张图里面的所有水平的边即可。
对于每条边,它所拥有的属性是:这条边的左右端点(的横坐标),这条边的高度(纵坐标),这条边属于矩形的上边还是下边(想想为什么保存这个属性)
刚刚计算中我们遇到两个蓝色矩形的一部分一眼就能看出这两个蓝色矩形的‘宽’是多少,用计算机怎么做到?
线段树华丽登场!
我们以整个图最左边的竖线作为区间左端点,最右边的竖线作为区间右端点,去维护这个区间的有效长度(即被覆盖的长度)
比如扫到第2条边的时候,有效长度就是两个蓝色矩形的宽之和。
这样,我们用扫描线去扫描每一条边的时候,都需要更新线段树的有效长度
是如何更新的呢?
如果扫到的这条边是某矩形的下边,则往区间插入这条线段
如果扫到的这条边是某矩形的上边,则往区间删除这条线段
为什么?自己试着模拟一下就不难发现:
因为我们是自下而上的扫这个图,扫到下边相当于刚刚进入一个矩形,扫到上边则是要离开一个矩形
利用线段树把每条边的有效长度找到了,也就是找到了每部分的所有矩形的总宽,那么高呢?
高就简单多了,对于所有的边,按照高度从小到大排列,那么矩形高就是每相邻边之间的高度差
给个例子:HDU 1542 Atlantis
然后看看用代码具体是怎么实现的:
ps: 特别说一下,关于上边和下边的标记,用-1标记下边,1标记上边是最合理的(想想为什么,提示:下边--删除,上边--插入)
这题横坐标略大,需要离散化处理
#define mem(a,x) memset(a,x,sizeof(a))
#include
#include
#include
#include
#include
#include
#include
#include
#include
说完了矩形面积,矩形周长的方法自然是类似的,但是周长的计算却更复杂些,看这张图:
周长可以分成两部分计算,横线和竖线,如图将所有彩色的横线加起来就是横向的所有长度了
然后可以采用竖直方向的扫描线将竖线的所有长度求出来
那么怎么计算横线的长度呢?
横线的长度 = 【现在这次总区间被覆盖的程度和上一次总区间被覆盖的长度之差的绝对值】
想想为什么要加绝对值(提示:下边--删除,上边--插入)
这样用自下而上和从左往右的两次扫描即可得到答案。
但是,这样的方法显得有些笨,有没有更高端的方法呢?
再看一张图:
看出什么了吗?这张图在上面那张图的基础上多了几条竖线。
我的意思是说,我们可以只做一次自下而上的扫描就把横线竖线都算出来!
竖线的算法和上面说的方法一样:【现在这次总区间被覆盖的程度和上一次总区间被覆盖的长度之差的绝对值】
竖线要怎么计算?
首先我们现在改一下线段树保存的属性,我们用如下信息记录线段树的节点:
1. l , r : 该节点代表的线段的左右端点坐标
2.len : 这个区间被覆盖的长度(即计算时的有效长度)
3.s : 表示这个区间被覆盖了几次
4. lc , rc : 标记这个节点的左右两个端点是否被覆盖(0表示没有,1表示有)
5.num :这个区间有多少条线段(这个区间被多少条线段覆盖)
这里的num涉及到竖线的计算,故解释一下,举几个例子:
若区间[0,10]被[1,2][4,5]覆盖,则num = 2
若区间[0,10]被[1,3][4,5]覆盖,则num = 1(两区间刚好连在一起)
若区间[0,10]被[1,5][2,6]覆盖,则num = 1(两区间连起来还是一段)
然后就可以计算竖线了:
竖线的长度 = 【下一条即将被扫到的横线的高度 - 现在扫到的横线的高度】*2*num
乘2是因为每条线段有两个端点;
看上图中棕色线段的竖线有4条,因为棕色的横线由2条线段组成
白色线段的竖线只有2条,因为白色的横线由1条线段组成(虽然这1条线段是由许多线段组合而成,但它依旧只算1条线段)
这样,依旧只扫一次就可以算出周长。
给个例子:HDU 1828Picture
代码实现:
#define mem(a,x) memset(a,x,sizeof(a))
#include
#include
#include
#include
#include
#include
#include
#include
#include
进阶:
HDU 1255 覆盖的面积(线段树+扫描线求面积【升级版】)
HDU 3642 Get The Treasury(线段树+扫描线求面积【再升级版(三维)】)
参考资料:
http://www.cnblogs.com/scau20110726/archive/2013/04/12/3016765.html
http://www.cnblogs.com/scau20110726/archive/2013/04/13/3018687.html