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

相关文章

Java如何根据word模板导出数据

《Java如何根据word模板导出数据》这篇文章主要为大家详细介绍了Java如何实现根据word模板导出数据,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... pom.XML文件导入依赖 <dependency> <groupId>cn.afterturn</groupId>

Oracle 通过 ROWID 批量更新表的方法

《Oracle通过ROWID批量更新表的方法》在Oracle数据库中,使用ROWID进行批量更新是一种高效的更新方法,因为它直接定位到物理行位置,避免了通过索引查找的开销,下面给大家介绍Orac... 目录oracle 通过 ROWID 批量更新表ROWID 基本概念性能优化建议性能UoTrFPH优化建议注

Python中Flask模板的使用与高级技巧详解

《Python中Flask模板的使用与高级技巧详解》在Web开发中,直接将HTML代码写在Python文件中会导致诸多问题,Flask内置了Jinja2模板引擎,完美解决了这些问题,下面我们就来看看F... 目录一、模板渲染基础1.1 为什么需要模板引擎1.2 第一个模板渲染示例1.3 模板渲染原理二、模板

Redis中6种缓存更新策略详解

《Redis中6种缓存更新策略详解》Redis作为一款高性能的内存数据库,已经成为缓存层的首选解决方案,然而,使用缓存时最大的挑战在于保证缓存数据与底层数据源的一致性,本文将介绍Redis中6种缓存更... 目录引言策略一:Cache-Aside(旁路缓存)策略工作原理代码示例优缺点分析适用场景策略二:Re

Pandas利用主表更新子表指定列小技巧

《Pandas利用主表更新子表指定列小技巧》本文主要介绍了Pandas利用主表更新子表指定列小技巧,通过创建主表和子表的DataFrame对象,并使用映射字典进行数据关联和更新,实现了从主表到子表的同... 目录一、前言二、基本案例1. 创建主表数据2. 创建映射字典3. 创建子表数据4. 更新子表的 zb

利用Python打造一个Excel记账模板

《利用Python打造一个Excel记账模板》这篇文章主要为大家详细介绍了如何使用Python打造一个超实用的Excel记账模板,可以帮助大家高效管理财务,迈向财富自由之路,感兴趣的小伙伴快跟随小编一... 目录设置预算百分比超支标红预警记账模板功能介绍基础记账预算管理可视化分析摸鱼时间理财法碎片时间利用财

如何在 Spring Boot 中实现 FreeMarker 模板

《如何在SpringBoot中实现FreeMarker模板》FreeMarker是一种功能强大、轻量级的模板引擎,用于在Java应用中生成动态文本输出(如HTML、XML、邮件内容等),本文... 目录什么是 FreeMarker 模板?在 Spring Boot 中实现 FreeMarker 模板1. 环

IDEA自动生成注释模板的配置教程

《IDEA自动生成注释模板的配置教程》本文介绍了如何在IntelliJIDEA中配置类和方法的注释模板,包括自动生成项目名称、包名、日期和时间等内容,以及如何定制参数和返回值的注释格式,需要的朋友可以... 目录项目场景配置方法类注释模板定义类开头的注释步骤类注释效果方法注释模板定义方法开头的注释步骤方法注

MySQL更新某个字段拼接固定字符串的实现

《MySQL更新某个字段拼接固定字符串的实现》在MySQL中,我们经常需要对数据库中的某个字段进行更新操作,本文就来介绍一下MySQL更新某个字段拼接固定字符串的实现,感兴趣的可以了解一下... 目录1. 查看字段当前值2. 更新字段拼接固定字符串3. 验证更新结果mysql更新某个字段拼接固定字符串 -

MySQL新增字段后Java实体未更新的潜在问题与解决方案

《MySQL新增字段后Java实体未更新的潜在问题与解决方案》在Java+MySQL的开发中,我们通常使用ORM框架来映射数据库表与Java对象,但有时候,数据库表结构变更(如新增字段)后,开发人员可... 目录引言1. 问题背景:数据库与 Java 实体不同步1.1 常见场景1.2 示例代码2. 不同操作