题目大意

一个数列,每次操作可以是将某区间数字都加上一个相同的整数,也可以是询问一个区间中所有数字的和。(这里区间指的是数列中连续的若干个数)对每次询问给出结果。

思路

对于区间的查找更新操作,可以考虑使用伸展树、线段树等数据结构。这里使用线段树来解决。需要注意的是,对于一个区间的增加操作,如果每次都走到叶子节点进行更新,则必定超时,因此lazy方法来解决。即如果能从当前节点获得所需要的信息,则不必走到子节点。

实现(c++)

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<stdio.h>
using namespace std;
#define MAX_COUNT 100005
#define MAX(a, b) a>b? a:b
#define MIN(a, b) a<b? a:b //查找两个相交的区间的并长度(即两个区间覆盖的总长度)
int Interval(int beg1, int end1, int beg2, int end2){
int tmp = MAX(end1, end2);
//注意不要 MAX(end1, end2) -MIN(beg1, beg2), 因为这样做在编译的时候宏定义展开,
//和三目运算符一块,优先级导致结果出错!!!! tmp -= MIN(beg1, beg2);
return (end1 - beg1 + end2 - beg2) - tmp + ;
} int gNumCount, gQueryCount; //总的数字的数目,总的查询次数 int gNumber[MAX_COUNT]; //线段树节点的数组 //线段树节点的定义
struct TreeNode{
int begin; //该节点覆盖区间的左边界
int end; //该节点覆盖区间的右边界 long long sum; //该节点覆盖区间的当前sum值,不包括 inc 可能增加的那些值(即实际的和应该为 sum + inc *(end - begin+1)
long long inc; //该节点覆盖区间中的那些点 应该被增加的值(注意是该区间的所有点)
}; struct TreeNode gTreeNodes[MAX_COUNT*]; //自底向上的更新,将左右子的 sum 值相加得到 该节点的sum值
void PushUp(int node_index){
int left_child = * node_index + ;
int right_child = * node_index + ;
gTreeNodes[node_index].sum = gTreeNodes[left_child].sum + gTreeNodes[right_child].sum;
} //自顶向下的更新操作,当该节点不能被查询区间全部覆盖,则需要向下走,去其子节点位置进行查询
//在查询之前,需要将 inc 值传递到子节点,同时该节点的sum值更新,inc值清零
void PushDown(int node_index){
TreeNode* node = gTreeNodes + node_index;
int left_child = * node_index + ;
int right_child = * node_index + ;
node->sum += node->inc*(node->end - node->begin + );
gTreeNodes[left_child].inc += node->inc;
gTreeNodes[right_child].inc += node->inc;
node->inc = ;
}
void BuildTree(int node_index, int beg, int end){
TreeNode* node = gTreeNodes + node_index;
node->inc = node->sum = ;
node->begin = beg;
node->end = end;
if (beg == end){
node->sum = gNumber[beg];
return;
}
int mid = (beg + end) / , left_child = * node_index + , right_child = * node_index + ;
BuildTree(left_child, beg, mid);
BuildTree(right_child, mid + , end); //自底向上更新
PushUp(node_index);
} void Add(int node_index, int beg, int end, long long c){
TreeNode* node = gTreeNodes + node_index;
int left_child = * node_index + , right_child = * node_index + , mid = (node->begin + node->end) / ;
if (node->begin > end || node->end < beg){
return;
}
//如果当前结点被查询区间全部覆盖,则不向下传递,直接将inc值增加
if (node->begin >= beg && node->end <= end){
node->inc += c;
return;
}
//如果节点不能被查询区间全部覆盖,则需要分裂节点向下传递,此时的sum值需要增加(inc * 两区间重合的长度)
//而同时 inc 值保持不变
node->sum += (Interval(node->begin, node->end, beg, end) * c);
int end1 = MIN(end, mid);
int beg1 = MAX(beg, mid + ); Add(left_child, beg, end1, c);
Add(right_child, beg1, end, c);
} long long QuerySum(int node_index, int beg, int end){
TreeNode* node = gTreeNodes + node_index;
// printf("node %d's sum = %d\n", node_index, gTreeNodes[node_index].sum);
// printf("node->beg = %d, node->end = %d, beg = %d, end = %d\n", node->begin, node->end, beg, end); int left_child = * node_index + , right_child = * node_index + , mid = (node->begin + node->end) / ;
long long sum = ;
if (node->begin > end || node->end < beg){
return sum;
}
if (node->begin >= beg && node->end <= end){
sum += (node->sum + node->inc*(node->end - node->begin + ));
}
else{
//向下分裂的更新操作
PushDown(node_index); int end1 = MIN(end, mid);
int beg1 = MAX(beg, mid + );
long long sum_left = QuerySum(left_child, beg, end1);
long long sum_right = QuerySum(right_child, beg1, end);
sum += (sum_left + sum_right);
}
return sum;
} int main(){
scanf("%d %d", &gNumCount, &gQueryCount);
for (int i = ; i < gNumCount; i++){
scanf("%d", gNumber + i);
}
BuildTree(, , gNumCount - ); char op;
int a, b;
long long c;
long long result;
for (int i = ; i < gQueryCount; i++){
getchar();
scanf("%c", &op);
if (op == 'C'){
scanf("%d %d %lld", &a, &b, &c);
Add(, a- , b-, c);
/*
for (int k = 0; k < 27; k++){
printf("node[%d]'s sum = %lld, inc = %lld\n", k, gTreeNodes[k].sum, gTreeNodes[k].inc);
}
*/
}
else if (op == 'Q'){
scanf("%d %d", &a, &b);
result = QuerySum(, a-, b-);
printf("%lld\n", result);
}
}
return ;
}

poj_3468 线段树的更多相关文章

  1. poj_3468线段树成段更新求区间和

    #include<iostream> #include<string.h> #include<cstdio> long long num[100010]; usin ...

  2. poj_3468,线段树成段更新

    参考自http://www.notonlysuccess.com/index.php/segment-tree-complete/ #include<iostream> #include& ...

  3. POJ_3468 A Simple Problem with Integers 【线段树区间查询+修改】

    一.题目 POJ3468 二.分析 裸的线段树区间查询+修改. 三.AC代码 #include <cstdio> #include <iostream> #include &l ...

  4. 线段树(成段更新) POJ 3468 A Simple Problem with Integers

    题目传送门 /* 线段树-成段更新:裸题,成段增减,区间求和 注意:开long long:) */ #include <cstdio> #include <iostream> ...

  5. bzoj3932--可持久化线段树

    题目大意: 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第 ...

  6. codevs 1082 线段树练习 3(区间维护)

    codevs 1082 线段树练习 3  时间限制: 3 s  空间限制: 128000 KB  题目等级 : 大师 Master 题目描述 Description 给你N个数,有两种操作: 1:给区 ...

  7. codevs 1576 最长上升子序列的线段树优化

    题目:codevs 1576 最长严格上升子序列 链接:http://codevs.cn/problem/1576/ 优化的地方是 1到i-1 中最大的 f[j]值,并且A[j]<A[i] .根 ...

  8. codevs 1080 线段树点修改

    先来介绍一下线段树. 线段树是一个把线段,或者说一个区间储存在二叉树中.如图所示的就是一棵线段树,它维护一个区间的和. 蓝色数字的是线段树的节点在数组中的位置,它表示的区间已经在图上标出,它的值就是这 ...

  9. codevs 1082 线段树区间求和

    codevs 1082 线段树练习3 链接:http://codevs.cn/problem/1082/ sumv是维护求和的线段树,addv是标记这歌节点所在区间还需要加上的值. 我的线段树写法在运 ...

随机推荐

  1. 从钉钉微应用定制化导航栏看如何实现Hybrid App开发框架

    钉钉是阿里的一款企业应用APP,里面提供了混合微应用的SDK,这其实最好的一种APP架构模式.微信公众号浏览器JSSDK也提供了类似功能特性,在在交互性上没有钉钉深入. http://ddtalk.g ...

  2. FusionCharts JavaScript API - Events 全局事件处理

    FusionCharts JavaScript API - Events 全局事件处理 Home > FusionCharts XT and JavaScript > API Refere ...

  3. CentOS6.5下安装Zabbix

    一.安装环境LAMP 这次安装zabbix全程使用yum进行安装,在于速度快,而且,出错的机会小,如果以后在生产环境上时,也可以先使用可以联网的机器进行安装,安装好以后将机器隔离,这样安装速度更快更方 ...

  4. 推断iframe里的页面载入完毕

    //推断iframe是否载入完毕,RMid为iframe的ID document.getElementById("RMid").onload = function () { ale ...

  5. MySQL 性能优化的最佳 20+ 条经验

    今天,数据库的操作越来越成为整个应用的性能瓶颈了,这点对于Web应用尤其明显.关于数据库的性能,这并不只是DBA才需要担心的事,而这更是我 们程序员需要去关注的事情.当我们去设计数据库表结构,对操作数 ...

  6. 【转】淘宝技术牛p博客整理

    转自:http://blog.csdn.NET/zdp072/article/details/19574793 淘宝技术委员会是由淘宝技术部高级技术人员组成的一个组织,共分为Java分会.C/C++分 ...

  7. 写给测试人员:不是所有的bug都需要修复

    用户往往对产品中各种各样的bug抱怨不已,而测试人员往往认为自己的职责就是揪出这些所有的bug并把它们全都修复.然而,这是一个误区.微软卓越测试工程总监Alan Page近日撰文,再次解释了有哪些bu ...

  8. 关于树莓派 BOOBS 安装之后的初级操作

    以安装OpenCV 3.1.0为例 上一篇我们利用Raspberry 官方提供的工具 BOOBS安装了Raspbian 操作系统,下面让我们看一下如何简单的配置raspbian操作系统. 从树莓派官方 ...

  9. 解决App can’t be opened because it is from an unidentified developer

    关闭设置 打开终端 输入sudo spctl --master-disable

  10. e645. 处理键盘事件

    You can get the key that was pressed either as a key character (which is a Unicode character) or as ...