线段树(Segment tree).docVIP

  1. 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
  2. 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  3. 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  4. 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  5. 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  6. 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  7. 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
线段树(Segment tree)

线段树(Segment tree) 德尔(n); 如果(C = = Q’)cout t [n] endl; } } 返回0; } 另一种功能上比较类似的数据结构:“树状数组它们有不少相似之处”: 针对点集的处理(添加、删除、查找); 相似的时空复杂度(logn时间,2n空间); 相似的编程复杂度(都比线段树简短得多); 因此,所有可以用树状数组解决的问题都可以用这个”点树”来解决,另外它还有以下好处: 更直观的转移; 同时支持自下而上和自上而下两种方向的查找和更新,而后者树状数组不支持,所以树状数组不提供某些功能,比如说O(logN)求点集中第K小数。 转]线段树2008-08-05 09:53 在一类问题中,我们需要经常处理可以映射在一个坐标轴上的一些固定线段,例如说映射在牛轴上的线段。由于线段是可以互相覆盖的,有时需要动态地取线段的并,例如取得并区间的总长度,或者并区间的个数等等。一个线段是对应于一个区间的,因此线段树也可以叫做区间树。 线段树的构造思想 线段树处理的是一定的固定线段,或者说这些线段是可以对应于有限个固定端点的。处理问题的时候,首先抽象出区间的端点,例如说N个端点Ti(1≤我≤N)。那么对于任何一个要处理的线段(区间)[ b ],来说,总可以找到相应的I,J,使得Ti =,TJ = B,1≤我≤J≤N.这样的转换就使得线段树上的区间表示为整数,通过映射转换,可以使得原问题实数个区间得到同样的处理。下图显示了一个能够表示[ 1 ]的线段树: 线段树是一棵二叉树,树中的每一个结点表示了一个区间[a,b]。每一个叶子节点上+ 1 = B,这表示了一个初等区间。对于每一个内部结点B-A>1,设根为[a,b ]的线段树为T(a,b),则进一步将此线段树分为左子树T(A,(A + B)/ 2),以及右子树T((a+b)/ 2,B),直到分裂为一个初等区间为止。 线段树的平分构造,实际上是用了二分的方法。线段树是平衡树,它的深度为日志(B-A)。 如果采用动态的数据结构来实现线段树,结点的构造可以用如下数据结构: Struct Tnode { int,e; 数组的个数; Tnode LeftChild,右孩子; }; 其中B和E表示了该区间为[ B、E ],算为一个计数器,通常记录覆盖到此区间的线段的个数。leftchild和分别是左右子树的根右孩子。 或者为了方便,我们也采用静态的数据结构。用数组B [ ],[ ] [ ] E,C,lson [ ],[ ]的人。设一棵线段树的根为诉那么B [V],E [V]就是它所表示区间的界。C [五]仍然用来作计数器。lson [V],RSON [V]分别表示了它的左儿子和右儿子的根编号。 注意,这只是线段树的基本结构。通常利用线段树的时候需要在每个结点上增加一些特殊的数据域,并且它们是随线段的插入删除进行动态维护的。这因题而异,同时又往往是解题的灵魂。 线段树处理数据的基本方法 线段树的最基本的建立,插入和删除的过程,以静态数据结构为例。 建立线段树(A,B): 设一个全局变量N,来记录一共用到了多少结点。开始n = 0。 空构建(a,b) { N←n + 1 V←n B [五]←一 E [V]←B C [五]←0 如果(b - A 1) { lson [v] ←n + 1 build (a,) rson [v] ←n + 1 build (b) } } 插入线段树: 将区间 [c, d] 插入线段树t (a, b), 并设t (a, b) 的根编号为v. void insert (c, d, v) { if (c≤b [v] f [v] ≤d) c [v] ←c [v] + 1; else if (c ) insert (c, d, lson [v]); if (d ) insert (c, d, rson [v]); } 对于此算法的解释: 如果 [c, d] 完全覆盖了当前线段, 那么显然该结点上的基数 (即覆盖线段数) 加1. 否则, 如果 [c, d] 不跨越区间中点, 就只对左树或者右树上进行插入. 否则, 在左树和右树上都要进行插入. 注意观察插入的路径, 一条待插入区间在某一个结点上进行 跨越, 此后两条子树上都要向下插入, 但是这种跨越不可能多次发生. 插入区间的时间复杂度是o (logn). 线段树删除: 在线段树上删除一个区间与插入的方法几乎是完全类似的. 将区间 [c, d] 删除于线段树t (a, b), 并设t (a, b) 的根编号为v. void delete (c, d, v) { if (c≤b [v] f [v] ≤d) c [v] ←c [v] - 1. else if (c ) delete (c, d, lson [v])

您可能关注的文档

文档评论(0)

jgx3536 + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

版权声明书
用户编号:6111134150000003

1亿VIP精品文档

相关文档