zoukankan      html  css  js  c++  java
  • 线段树 (扫描线)

      这里用HDU的1542题作为例子,一个经典的扫描线题目,计算矩形并的和。

      首先介绍扫描线,就是一根假想的线,从左到右的一条竖线扫描过去。

      扫描线可以用来填充多边形,具体请看 http://blog.csdn.net/orbit/article/details/7368996  写的很好。

      然后就是扫描线在这个题里的应用。

      

      计算这两个矩形的面积等价于计算红色,绿色,蓝色三块的面积的和。

      所以说从左到右的扫描线扫描的时候,线段树维护的是区间内被覆盖的总长度,然后乘以两条扫描线之间的距离,就是一块的面积,然后更新线段树,以此论推,得到的就是总面积了。

      不过这个题要注意三个地方:

      1,离散化,每条边的坐标是浮点数,要进行离散化,要记得去重。

      2,就是区间覆盖的长度计算问题,这里我纠结了老久,首先如果按照点来覆盖的话对区间2,8覆盖其实是对1,5和6,10进行覆盖(如果线段树左右子树的区间是1,5和6,10的话,那么5和6中间那一块就没有被计算上,就出问题了。。。

      所以说不能按照点来覆盖,应该把每个点当作是表示这个点到下一个点之间的这条线段,这样就不会漏掉中间那一个了。

      如果是表示线段的话,那么如果有10个点,那么有9条线段,所以线段树是1到9(而不是10),每次计算长度的时候记得右边要加一。

      3,就是线段树的更新问题,每加一条线段或者减一条线段的话,并不是简单的覆盖就好,而是应该记录这个区间被覆盖了几次,直到减到零的时候才是没有线段在覆盖这个区间。这样的话每加一条线段就是对线段的区间+1,去一条线段就是-1。

      不过这样的话pushDown就不好用了,因为如果把一个节点的COL值推到下面两个之后,之后如果再覆盖,这个节点的COL变成-1了,那就不好算这个节点维护的值了。所以根据杭电大神的写法,不用pushDown,而是在计算pushUP的时候,如果这个节点已经被覆盖为大于0的就直接计算(因为这个节点表示的整个区间都被覆盖了,直接右面减去左面就是了。),否则才求助于下面的两个节点。而且,询问只是询问节点1的值,所以不会出错。

      近来又看到了一个大神的线段树写法,他用了pushDown,但是也对pushUP进行了修改,从而保证不会让某一个节点的值为零,虽说速度不会快,但是可扩展性要高很多,因为可以让有的矩形第一条边为-1,第二条边为1(HDU  3265)这个题。

      修改的代码我在HDU  3265和HDU  1542中都贴出来了。

      差不多就这三个了,水平有限,敬请谅解。

    例题:

      HDU  1542  题解请单击

      HDU  1828  题解请单击

      HDU  3265  题解请单击

  • 相关阅读:
    jQuery return false
    js获取拼音
    jQuery.getScript加载同域JS
    清除浮动,clearfix的bug
    chrome本地文件加载跨域请求
    中间容器宽度自适应的三栏布局
    mac安装nginx
    弹出一次“设为主页”和“加入收藏”代码
    未在本地计算机上注册“Microsoft.Jet.OLEDB.4.0”提供程序
    IIS7/IIS7.5下轻松配置PHP利器(微软PHP Manager for IIS 7)
  • 原文地址:https://www.cnblogs.com/whywhy/p/4214353.html
Copyright © 2011-2022 走看看