博客
关于我
Vijos-P1103题解【线段树】
阅读量:437 次
发布时间:2019-03-06

本文共 699 字,大约阅读时间需要 2 分钟。

一条马路从数轴0到L,每个位置0,1,2,…,L都有一棵树。现需要将一些区间的树清除,区间可能有交集,求清除后还剩下多少棵树。

思路分析

第一感觉,可以用数组a[10000]来模拟马路上的树,1表示有树,每次操作则将对应区间所有点改为0,最终统计为1的点,但时间复杂度O(L*M),这明显不是最优的解法。

再仔细一想,每次都是对一个区间进行操作,而且对每一个点的操作都是相同的,所以考虑是否可以批量操作。

对于批量操作区间有线段树,树状树组等算法,本文讨论用线段树来解,树状树组算法以后会详细介绍。

那么问题来了,为什么会想到用线段树来解呢?即线段树能解决哪类问题?

线段树关键点:大区间的操作及结果等价于两个相邻的子区间操作及结果。

每次对马路上的树进行区间操作,如移除区间[a,b]上的树,也等价于移除区间[a,k],k+1,b上的树。

并且当区间上的树已经移除后,再重复移除也对最终结果无影响。

如上图,树中每个节点表示一段区间,每个节点需要记录如下关键信息:

left: 区间左起点

right: 区间右终点

count: 区间还剩下的树,初始为right-left+1;

1)更新树时,如果区间在需要操作的范围内,则将区间所有树清除,即count=0,直接返回,不需要再去清除所有子节点。

2)父节点同时更新count值,father.count=lson.count+right.count;

  当父节点已经为0,则说明该区间已经全部被清除,此时左右子节点之和可能不等于0,看1)中并没有去清除子节点;

  所以父节点还是保持0

最终剩下的树即为根节点中的树,即root.count。

转载地址:http://jfcyz.baihongyu.com/

你可能感兴趣的文章
Objective-C实现跳跃游戏的动态编程自上而下的方法算法(附完整源码)
查看>>
Objective-C实现跳跃游戏的动态编程自下而上的方法的算法(附完整源码)
查看>>
Objective-C实现跳跃游戏的贪婪方法的算法(附完整源码)
查看>>
Objective-C实现车牌识别系统(附完整源码)
查看>>
Objective-C实现转置加解密文件算法(附完整源码)
查看>>
Objective-C实现转置密码算法(附完整源码)
查看>>
Objective-C实现软键盘功能(附完整源码)
查看>>
Objective-C实现输入两个浮点数,输出它们中的大数(附完整源码)
查看>>
Objective-C实现输出不同类型所占的字节数(附完整源码)
查看>>
Objective-C实现辗转相除法(附完整源码)
查看>>
Objective-C实现辗转相除法算法(附完整源码)
查看>>
Objective-C实现边缘检测Canny(附完整源码)
查看>>
Objective-C实现边缘检测Canny(附完整源码)
查看>>
Objective-C实现近邻传播算法(附完整源码)
查看>>
Objective-C实现返回 Collatz 序列及其任意正整数的长度算法(附完整源码)
查看>>
Objective-C实现返回 n^pow 的幂位和算法(附完整源码)
查看>>
Objective-C实现返回2个字符串的替代字符串排列算法(附完整源码)
查看>>
Objective-C实现返回一个包含所有节点邻居的数组算法(附完整源码)
查看>>
Objective-C实现返回数字的二进制表示中使用的位数bitLength算法(附完整源码)
查看>>
Objective-C实现进度条(附完整源码)
查看>>