线段树成段更新操作及Lazy思想(POJ3468解题报告)

标签: treequerybuildn2cstruct
2011-11-03 20:37 5756人阅读 评论(0) 收藏 举报
 分类:
POJ解题报告(5)  数据结构(21) 

版权声明:本文为博主原创文章,未经博主允许不得转载。

就直接那POJ上面的例题来说吧,http://poj.org/problem?id=3468。

此题题意很好懂:

给你N个数,Q个操作,操作有两种,‘Q a b ’是询问a~b这段数的和,‘C a b c’是把a~b这段数都加上c。

需要用到线段树的,update:成段增减,query:区间求和

介绍Lazy思想:lazy-tag思想,记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。

在此通俗的解释我理解的Lazy意思,比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,它的节点标记为rt,这时tree[rt].l == a && tree[rt].r == b 这时我们可以一步更新此时rt节点的sum[rt]的值,sum[rt] += c * (tree[rt].r - tree[rt].l + 1),注意关键的时刻来了,如果此时按照常规的线段树的update操作,这时候还应该更新rt子节点的sum[]值,而Lazy思想恰恰是暂时不更新rt子节点的sum[]值,到此就return,直到下次需要用到rt子节点的值的时候才去更新,这样避免许多可能无用的操作,从而节省时间 。

下面通过具体的代码来说明之。(此处的函数名和宏学习了小HH的代码风格)

在此先介绍下代码中的函数说明:

#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1

宏定义左儿子lson和右儿子rson,貌似用宏的速度要慢。

PushUp(rt):通过当前节点rt把值递归向上更新到根节点

PushDown(rt):通过当前节点rt递归向下去更新rt子节点的值

rt表示当前子树的根(root),也就是当前所在的结点

[cpp] view plaincopy print?
  1. __int64 sum[N<<2],add[N<<2];
  2. struct Node
  3. {
  4. int l,r;
  5. int mid()
  6. {
  7. return (l+r)>>1;
  8. }
  9. } tree[N<<2];

这里定义数据结构sum用来存储每个节点的子节点数值的总和,add用来记录该节点的每个数值应该加多少

tree[].l tree[].r分别表示某个节点的左右区间,这里的区间是闭区间

下面直接来介绍update函数,Lazy操作主要就是用在这里

[cpp] view plaincopy print?
  1. void update(int c,int l,int r,int rt)//表示对区间[l,r]内的每个数均加c,rt是根节点
  2. {
  3. if(tree[rt].l == l && r == tree[rt].r)
  4. {
  5. add[rt] += c;
  6. sum[rt] += (__int64)c * (r-l+1);
  7. return;
  8. }
  9. if(tree[rt].l == tree[rt].r) return;
  10. PushDown(rt,tree[rt].r - tree[rt].l + 1);
  11. int m = tree[rt].mid();
  12. if(r <= m) update(c,l,r,rt<<1);
  13. else if(l > m) update(c,l,r,rt<<1|1);
  14. else
  15. {
  16. update(c,l,m,rt<<1);
  17. update(c,m+1,r,rt<<1|1);
  18. }
  19. PushUp(rt);
  20. }

if(tree[rt].l == l && r == tree[rt].r) 这里就是用到Lazy思想的关键时刻 正如上面说提到的,这里首先更新该节点的sum[rt]值,然后更新该节点具体每个数值应该加多少即add[rt]的值,注意此时整个函数就运行完了,直接return,而不是还继续向子节点继续更新,这里就是Lazy思想,暂时不更新子节点的值。

那么什么时候需要更新子节点的值呢?答案是在某部分update操作的时候需要用到那部分没有更新的节点的值的时候,这里可能有点绕口。这时就掉用PushDown()函数更新子节点的数值。

[cpp] view plaincopy print?
  1. void PushDown(int rt,int m)
  2. {
  3. if(add[rt])
  4. {
  5. add[rt<<1] += add[rt];
  6. add[rt<<1|1] += add[rt];
  7. sum[rt<<1] += add[rt] * (m - (m>>1));
  8. sum[rt<<1|1] += add[rt] * (m>>1);
  9. add[rt] = 0;//更新后需要还原
  10. }
  11. }

PushDown就是从当前根节点rt向下更新每个子节点的值,这段代码读者可以自己好好理解,这也是Lazy的关键。

接着就是update操作的三个if语句了,这里我曾经一直不理解,多亏nyf队友的指点,借此感谢之。

下面再解释query函数,也就是用这个函数来求区间和

[cpp] view plaincopy print?
  1. __int64 query(int l,int r,int rt)
  2. {
  3. if(l == tree[rt].l && r == tree[rt].r)
  4. {
  5. return sum[rt];
  6. }
  7. PushDown(rt,tree[rt].r - tree[rt].l + 1);
  8. int m = tree[rt].mid();
  9. __int64 res = 0;
  10. if(r <= m) res += query(l,r,rt<<1);
  11. else if(l > m) res += query(l,r,rt<<1|1);
  12. else
  13. {
  14. res += query(l,m,rt<<1);
  15. res += query(m+1,r,rt<<1|1);
  16. }
  17. return res;
  18. }

第一个if还是区间的判断和前面update的一样,到这里就可以知道答案了,所以就直接return。

接下来的查询就需要用到rt子节点的值了,由于我们用了Lazy操作,这段的数值还没有更新,因此我们需要调用PushDown函数去更新之,满足if(add[rt])就说明还没有更新。

到这里整个Lazy思想就算介绍结束了,可能我的语言组织不是很好,如果有不理解的地方可以给我留言,我再解释大家的疑惑。

PS:今天总算是对线段树入门了。

这里推荐一下,完全版线段树网址

下面贴出POJ3468完整的代码http://www.notonlysuccess.com/index.php/segment-tree-complete/,这里面有很飘逸的线段树代码,表示其update和query写的很巧妙,代码量也比较少,大家可以去学习。

[cpp] view plaincopy print?
  1. #include <iostream>
  2. #include <cstdio>
  3. using namespace std;
  4. const int N = 100005;
  5. #define lson l,m,rt<<1
  6. #define rson m+1,r,rt<<1|1
  7. __int64 sum[N<<2],add[N<<2];
  8. struct Node
  9. {
  10. int l,r;
  11. int mid()
  12. {
  13. return (l+r)>>1;
  14. }
  15. } tree[N<<2];
  16. void PushUp(int rt)
  17. {
  18. sum[rt] = sum[rt<<1] + sum[rt<<1|1];
  19. }
  20. void PushDown(int rt,int m)
  21. {
  22. if(add[rt])
  23. {
  24. add[rt<<1] += add[rt];
  25. add[rt<<1|1] += add[rt];
  26. sum[rt<<1] += add[rt] * (m - (m>>1));
  27. sum[rt<<1|1] += add[rt] * (m>>1);
  28. add[rt] = 0;
  29. }
  30. }
  31. void build(int l,int r,int rt)
  32. {
  33. tree[rt].l = l;
  34. tree[rt].r = r;
  35. add[rt] = 0;
  36. if(l == r)
  37. {
  38. scanf("%I64d",&sum[rt]);
  39. return ;
  40. }
  41. int m = tree[rt].mid();
  42. build(lson);
  43. build(rson);
  44. PushUp(rt);
  45. }
  46. void update(int c,int l,int r,int rt)
  47. {
  48. if(tree[rt].l == l && r == tree[rt].r)
  49. {
  50. add[rt] += c;
  51. sum[rt] += (__int64)c * (r-l+1);
  52. return;
  53. }
  54. if(tree[rt].l == tree[rt].r) return;
  55. PushDown(rt,tree[rt].r - tree[rt].l + 1);
  56. int m = tree[rt].mid();
  57. if(r <= m) update(c,l,r,rt<<1);
  58. else if(l > m) update(c,l,r,rt<<1|1);
  59. else
  60. {
  61. update(c,l,m,rt<<1);
  62. update(c,m+1,r,rt<<1|1);
  63. }
  64. PushUp(rt);
  65. }
  66. __int64 query(int l,int r,int rt)
  67. {
  68. if(l == tree[rt].l && r == tree[rt].r)
  69. {
  70. return sum[rt];
  71. }
  72. PushDown(rt,tree[rt].r - tree[rt].l + 1);
  73. int m = tree[rt].mid();
  74. __int64 res = 0;
  75. if(r <= m) res += query(l,r,rt<<1);
  76. else if(l > m) res += query(l,r,rt<<1|1);
  77. else
  78. {
  79. res += query(l,m,rt<<1);
  80. res += query(m+1,r,rt<<1|1);
  81. }
  82. return res;
  83. }
  84. int main()
  85. {
  86. int n,m;
  87. while(~scanf("%d %d",&n,&m))
  88. {
  89. build(1,n,1);
  90. while(m--)
  91. {
  92. char ch[2];
  93. scanf("%s",ch);
  94. int a,b,c;
  95. if(ch[0] == 'Q')
  96. {
  97. scanf("%d %d", &a,&b);
  98. printf("%I64d\n",query(a,b,1));
  99. }
  100. else
  101. {
  102. scanf("%d %d %d",&a,&b,&c);
  103. update(c,a,b,1);
  104. }
  105. }
  106. }
  107. return 0;
  108. }

uscao 线段树成段更新操作及Lazy思想(POJ3468解题报告)相关推荐

  1. HDOJ 1698 Just a Hook(线段树成段更新)

    题意: 屠夫的钩子区间是1~n,每段可能由铜,银,金组成,价值分别为1,2,3,进行一系列的更新之后,求钩子的总价值. 思路: 线段树的成段更新:要设置一个临时的线段树,每次更新的时候把更新段的值放在 ...

  2. HDU 3974 Assign the task(dfs时间戳+线段树成段更新)

    题意:给定点的上下级关系,规定假设给i分配任务a.那么他的全部下属.都停下手上的工作,開始做a. 操作 T x y 分配x任务y,C x询问x的当前任务: Sample Input 1 5 4 3 3 ...

  3. HDU 6203 ping ping ping lca 线段树成段更新

    题目链接:HDU 6203 ping ping ping Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K ( ...

  4. Wikioi 1081 线段树成段更新单点查询

    线段树练习飘逸的写法,自从自己改成这样的写法之后,线段树就没再练过,如今最终练得上了. 由于这里查询仅仅是查询了叶子结点,所以pushUp函数就用不上了,只是我没去掉之前是3ms.去掉之后反而变成4m ...

  5. UVA 12501 Bulky process of bulk reduction ——(线段树成段更新)

    和普通的线段树不同的是,查询x~y的话,给出的答案是第一个值的一倍加上第二个值的两倍一直到第n个值的n倍. 思路的话,就是关于query和pushup的方法.用一个新的变量sum记录一下这个区间里面按 ...

  6. POJ3468 A Simple Problem with Integers【线段树 成段更新+求和 lazy标志】

    用longlong替换__int64也成. #define LL long long 输入输出用%lld Problem: 3468   User: qq1203456195 Memory: 4284 ...

  7. HDU 3397 Sequence operation 线段树 成段更新 区间合并

    比较综合的题. 两个标记  setv,xorr.setv的优先级高于xorr,当一个节点获得一个setv时,他之前的xorr要清除. //#pragma comment(linker, "/ ...

  8. hdu 1698 线段树成段更新

    这么重要的数据结构还是得学啊.仿notonlysuccess的代码风格 #include <cstdio> #include <algorithm> #include < ...

  9. UESTC-1057 秋实大哥与花(线段树+成段加减+区间求和)

    秋实大哥与花 Time Limit: 3000/1000MS (Java/Others)     Memory Limit: 65535/65535KB (Java/Others) Submit St ...

最新文章

  1. apache 编译php mysql_记一次痛苦的过程-------源码编译安装apache+php5+mysql
  2. 如何将Android的AOSP仓库放置到自己的gitlab服务器上?
  3. 对于出差的看法_我对于挑选背包的一些小建议
  4. 重写ScrollView实现两个ScrollView的同步滚动显示
  5. android.graphics包中的一些类的使用
  6. 每日一句090516
  7. baidumap vue 判断范围_vue 经典面试题+答案
  8. 国内首款安全软件开源,金山卫士源码共享
  9. 餐饮智能化:餐饮机器人正当时餐饮机器人的喜与忧
  10. 数字化时代,需要数据思维!
  11. PyQt5教程(七)——实现QQ登录界面(一、Qt Designer创建界面,Eric6创建项目)
  12. 程序员到创业,成长之路的技能分享
  13. Linux配置文件-limits.conf
  14. 三、pytest接口自动化之pytest中setup/teardown,setup_class/teardown_class讲解
  15. Web测试中定位bug方法
  16. 互联网快讯:龙佰集团冲刺港交所;极米Z6X Pro、极米H3S持续热销;京东物流调集3246人增援上海
  17. jquery实现全选
  18. 与钩React过度-实际操作
  19. 深度学习-BP神经网络(python3代码实现)
  20. 喜 欢 和 爱 的 区 别

热门文章

  1. 通过python利用哈希值实现比较两个文件的一致性
  2. 在批评中改变自己,才能真正取得进步
  3. 在idea使用maven工程建立web项目时,启动Tomcat访问不到项目首页。
  4. LeetCode简单题之检查是否区域内所有整数都被覆盖
  5. 高通为何46亿美元ADAS Veoneer Arriver
  6. vue 拓扑组件_Authing 登录组件优化实践解析
  7. HarmonyOS 使用DevEcoStudio创建一个xml布局以及引用布局
  8. 小菜鸟与后台对接接口下来的感受
  9. 7.26-Codeforces Round #372 (Div. 2)
  10. Pandoc PDF 中文