poj3468(线段树成段更新模板题)

2024-09-09 17:32

本文主要是介绍poj3468(线段树成段更新模板题),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和

下面的介绍是下解题思路:

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

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

代码如下:

#include<iostream>
#include<algorithm>
#include<cstring>
#include<stack>
#include<queue>
#include<set>
#include<map>
#include<stdio.h>
#include<stdlib.h>
#include<ctype.h>
#include<time.h>
#include<math.h>#define ll __int64
#define N 100005
#define inf 0x7ffffff
#define eps 1e-9
#define pi acos(-1.0)
#define P system("pause")
using namespace std;
struct Node
{int l,r;ll sum,add;
}tree[N<<2];
ll a[N];
void PushUp(int o)//用于更新根节点
{tree[o].sum = tree[2*o].sum + tree[2*o+1].sum;
}
void build(int o,int l,int r)
{tree[o].l = l;tree[o].r = r;tree[o].add = 0;if(l == r){tree[o].sum = a[l];return;}int m = (l+r)/2;build(2*o,l,m);build(2*o+1,m+1,r);PushUp(o);
}
void PushDown(int o)//用于更新子节点
{int m = (tree[o].r-tree[o].l+1);if( tree[o].add){tree[2*o].add += tree[o].add;tree[2*o+1].add += tree[o].add;tree[2*o].sum += tree[o].add*(m - (m>>1));tree[2*o+1].sum += tree[o].add*(m>>1);tree[o].add = 0;}
}
void update(int o,int x,int y,int v)
{if(x == tree[o].l && tree[o].r == y){tree[o].add += v;//这个标记用来延迟更新tree[o].sum += (__int64)v*(y-x+1);return ;}if(tree[o].l == tree[o].r) return;PushDown(o);int m = (tree[o].r+tree[o].l)/2;if(y <= m)update(2*o,x,y,v);else if(x > m)update(2*o+1,x,y,v);else {update(2*o, x, m, v);update(2*o+1, m+1, y, v);}PushUp(o);
}
ll query(int o,int x,int y)
{if(tree[o].l == x && tree[o].r == y)return tree[o].sum;PushDown(o);ll res = 0;int m = (tree[o].l+tree[o].r)/2;if(y <= m) res += query(2*o,x,y);else if(x > m) res += query(2*o+1,x,y);else {res += query(2*o,x,m);res += query(2*o+1,m+1,y);}return res;
}
int main()
{
//freopen("input.txt","r",stdin);
//freopen("output.txt","w",stdout);int n,m;while(scanf("%d%d",&n,&m) != EOF){int i;for(i = 1; i <= n; i++)scanf("%I64d",&a[i]);build(1,1,n);char str[5];while(m--){scanf("%s",str);int x,y,z;if(str[0] == 'C'){scanf("%d%d%d",&x,&y,&z);update(1,x,y,z);}if(str[0] == 'Q'){scanf("%d%d",&x,&y);printf("%I64d\n",query(1,x,y));}}}return 0;
}

最后还有几点补充,

在update函数中,if(tree[o].l == x && y == tree[o].r) 这里就是用到Lazy思想的关键时刻 正如上面说提到的,这里首先更新该节点的sum[o]值,然后更新该节点具体每个数值应该加多少即add[o]的值,注意此时整个函数就运行完了,直接return,而不是还继续向子节点继续更新,这里就是Lazy思想,暂时不更新子节点的值。那么什么时候需要更新子节点的值呢?答案是在某部分update操作的时候需要用到那部分没有更新的节点的值的时候,这时就掉用PushDown()函数更新子节点的数值。


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

这篇关于poj3468(线段树成段更新模板题)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1151817

相关文章

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

Linux Mint Xia 22.1重磅发布: 重要更新一览

《LinuxMintXia22.1重磅发布:重要更新一览》Beta版LinuxMint“Xia”22.1发布,新版本基于Ubuntu24.04,内核版本为Linux6.8,这... linux Mint 22.1「Xia」正式发布啦!这次更新带来了诸多优化和改进,进一步巩固了 Mint 在 Linux 桌面

基于Java实现模板填充Word

《基于Java实现模板填充Word》这篇文章主要为大家详细介绍了如何用Java实现按产品经理提供的Word模板填充数据,并以word或pdf形式导出,有需要的小伙伴可以参考一下... Java实现按模板填充wor编程d本文讲解的需求是:我们需要把数据库中的某些数据按照 产品经理提供的 word模板,把数据

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R

Ubuntu 24.04 LTS怎么关闭 Ubuntu Pro 更新提示弹窗?

《Ubuntu24.04LTS怎么关闭UbuntuPro更新提示弹窗?》Ubuntu每次开机都会弹窗提示安全更新,设置里最多只能取消自动下载,自动更新,但无法做到直接让自动更新的弹窗不出现,... 如果你正在使用 Ubuntu 24.04 LTS,可能会注意到——在使用「软件更新器」或运行 APT 命令时,

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

hdu1689(线段树成段更新)

两种操作:1、set区间[a,b]上数字为v;2、查询[ 1 , n ]上的sum 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdl

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :