poj3468 A Simple Problem with Integers(线段树模板 功能:区间增减,区间求和)[通俗易懂]

poj3468 A Simple Problem with Integers(线段树模板 功能:区间增减,区间求和)

大家好,又见面了,我是全栈君。

转载请注明出处:http://blog.csdn.net/u012860063

Description

You have N integers, A1A2, … , AN. You need to deal with two kinds of operations. One type of operation is to add some given number to each number in a given interval. The other is to ask for the sum of numbers in a given interval.

Input

The first line contains two numbers N and Q. 1 ≤ N,Q ≤ 100000.
The second line contains N numbers, the initial values of A1A2, … , AN. -1000000000 ≤ Ai ≤ 1000000000.
Each of the next Q lines represents an operation.
“C a b c” means adding c to each of AaAa+1, … , Ab. -10000 ≤ c ≤ 10000.
“Q a b” means querying the sum of AaAa+1, … , Ab.

Output

You need to answer all Q commands in order. One answer in a line.

Sample Input

10 5
1 2 3 4 5 6 7 8 9 10
Q 4 4
Q 1 10
Q 2 4
C 3 6 3
Q 2 4

Sample Output

4
55
9
15

Hint

The sums may exceed the range of 32-bit integers.

Source

代码例如以下:

//线段树功能:update:成段增减 query:区间求和
//此题为Poj 3468 代码

#include <cstdio>
#include <algorithm>
using namespace std;
#define lson l , mid , rt << 1
#define rson mid + 1 , r , rt << 1 | 1
//lson和rson分辨表示结点的左儿子和右儿子
//rt表示当前子树的根(root),也就是当前所在的结点
#define LL long long
const int maxn = 111111;
//maxn是题目给的最大区间,而节点数要开4倍,确切的来说节点数要开大于maxn的最小2x的两倍
LL add[maxn<<2];//用来标记每一个节点,为0则表示没有标记,否则为标记。
LL sum[maxn<<2];//求和
void PushUp(int rt) //把当前结点的信息更新到父结点
{
	sum[rt] = sum[rt<<1] + sum[rt<<1|1];
}
void PushDown(int rt,int len)//把当前结点的信息更新给儿子结点,len为分区间长度
 {//对某一个区间进行改变,假设被标记了,在查询的时候就得把改变传给子节点,由于查询的并不一定是当前区间  
	if (add[rt]) //已经标记过。该区间被改变过 
	{
		//由于rt的儿子节点可能被多次延迟标记。而且rt的儿子节点的延迟标记没有向rt的孙子节点移动,所以用“+=” 
		add[rt<<1] += add[rt];
		add[rt<<1|1] += add[rt];
		/*此处用add[rt]乘以区间长度,不是add[rt<<1], 由于rt的儿子节点假设被多次标记,之前被标记时, 
          就已经对sum[rt<<1]更新过了。

*/ sum[rt<<1] += add[rt] * (len - (len >> 1));//更新左儿子的和 sum[rt<<1|1] += add[rt] * (len >> 1);//更新右儿子的和 add[rt] = 0;//将标记向儿子节点移动后。父节点的延迟标记去掉 }}void build(int l,int r,int rt) { add[rt] = 0;//初始化为全部结点未被标记 if (l == r) { scanf("%lld",&sum[rt]); return ; } int mid = (l + r) >> 1; build(lson); build(rson); PushUp(rt);}void update(int L,int R,int c,int l,int r,int rt) { if (L <= l && r <= R) { add[rt] += c; sum[rt] += (LL)c * (r - l + 1);//更新代表某个区间的节点和,该节点不一定是叶子节点 return ; } /*当要对被延迟标记过的这段区间的儿子节点进行更新时,先要将延迟标记向儿子节点移动 当然,假设一直没有对该段的儿子节点更新,延迟标记就不须要向儿子节点移动,这样就使 更新操作的时间复杂度仍为O(logn),也是使用延迟标记的原因。 */ PushDown(rt , r - l + 1);//向下传递 int mid = (l + r) >> 1; if (L <= mid) update(L , R , c , lson);//更新左儿子 if (mid < R) update(L , R , c , rson);//更新右儿子 PushUp(rt);//向上传递更新和}LL query(int L,int R,int l,int r,int rt) { if (L <= l && r <= R) { return sum[rt]; }//要取rt子节点的值时。也要先把rt的延迟标记向下移动 PushDown(rt , r - l + 1); int mid = (l + r) >> 1; LL ret = 0; if (L <= mid) ret += query(L , R , lson); if (mid < R) ret += query(L , R , rson); return ret;}int main() { int N , Q; scanf("%d%d",&N,&Q);//N为节点数 build(1 , N , 1); //建树 while (Q--)//Q为询问次数 { char op[2]; int a , b , c; scanf("%s",op); if (op[0] == 'Q') { scanf("%d%d",&a,&b); printf("%lld\n",query(a , b , 1 , N , 1)); } else { scanf("%d%d%d",&a,&b,&c);//c为区间a到b添加的值 update(a , b , c , 1 , N , 1); } } return 0;}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/116347.html原文链接:https://javaforall.net

(0)
上一篇 2022年1月22日 下午6:00
下一篇 2022年1月22日 下午7:00


相关推荐

  • Socket 编程原理

    Socket 编程原理目录socket编程基本概念协议TCPUDPDNSICMPHTTPHTTPS编程流程socket函数socket编程基本概念socket编程即计算机网络编程,目的是使两台主机能够进行远程连接,既然要使两者产生联系,那么就要有至少一个信息发送端和一个信息接收端,因此形成了现在绝大多数socket编程都会用到的C/S架构(Client[客户端]/Server[服务端]),最典型的应用就是web服务器/客户端。在Unix/Linux中执行任何形式的I/O操作(比如网络连接)时,都是在读取

    2022年10月18日
    6
  • Java 工厂模式

    Java 工厂模式简单工厂模式详解简单工厂模式用来定义一个工厂类,它可以根据参数的不同返回不同类的实例,被创建的实例通常都具有共同的父类。因为在简单工厂模式中用于创建实例的方法是静态方法,因此简单工厂模式又被称为静态工厂方法模式,它属于类创建型模式。简单工厂模式的要点在于,当我们需要什么,只需要传入一个正确的参数,就可以获取我们所需要的对象,而无需知道其创建细节。简单工厂模式结构比较简单,其核心是工厂类的设计,其机构如图所示:在简单工厂模式结构图中包含如下几个角色。Factory(工厂角色):工厂角色即工厂类,它

    2022年7月20日
    24
  • IDEA 2020 3.3激活码_通用破解码

    IDEA 2020 3.3激活码_通用破解码,https://javaforall.net/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

    2022年3月15日
    408
  • 在ubuntu安装的软件在哪里找_ubuntu如何安装gcc编译器

    在ubuntu安装的软件在哪里找_ubuntu如何安装gcc编译器在ubuntu安装vscode和可视化的代码跟踪调试在ubuntu安装vscode一、命令安装1.在网页下载deb安装包:https://code.visualstudio.com/Download2.在命令行安装:3.在命令行执行:二、汉化在ubuntu中用vscode编译调试C\C++一、安装插件二、编译运行程序在ubuntu安装vscode一、命令安装1.在网页下载deb安装包:https://code.visualstudio.com/Download2.在命令行安装:

    2025年12月15日
    6
  • 实例与数据库的区别_mysql数据库实例是什么

    实例与数据库的区别_mysql数据库实例是什么mysql目前是开源界应用最为广泛的数据库软件了。相对于重量级的商业产品如oracle、DB2、SQLServer等,Mysql最大的特点就是开源免费。个人用户可以down一个下来,自己搭个网站玩玩。大型互联网企业诸如阿里、网易之类的也可以针对mysql做mysql集群和存储引擎的开发。今天主要是想解释一下mysql体系中,数据库和数据库实例的概念。很多人都在用mysql,也有很多人认为数据库就…

    2022年10月21日
    5
  • Win10怎么添加开机启动项?Win10添加开机自动运行软件三种方法

    Win10怎么添加开机启动项?Win10添加开机自动运行软件三种方法Win10 管理开机启动项的方法相信大家已经非常熟悉 msconfig 命令各系统都通用 那么很多用户发觉 Win10 和 Win7XP 等系统不同 没有启动文件夹 那么我们怎么添加开机启动项呢 如晨软件或程序没有开机启动设置的话 是的 在 Win10 中添加开机启动项虽然麻烦了些 但是还是可以设置的 下面小编就分享几种方法 方法一 开机启动文件夹 1 我们打开文件夹 C Users 用户 Administ

    2026年3月26日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号