Poj-P3468题解【线段树】
本文为原创,转载请注明:http://www.cnblogs.com/kylewilson/
题目出处:
http://poj.org/problem?id=3468
题目描述:
给N个数A1, A2, ... , AN. 你需要处理两种操作,一种操作是在一个区间上每个数都增加一个数,别一种操作是查询一个区间所有数的和
输入
第一行两个数N, Q, 1 ≤ N,Q ≤ 100000
第二行N个数,数组初始值A1, A2, ... , AN. -1000000000 ≤ Ai ≤ 1000000000.
接着Q行,表示Q次操作"C a b c" 表示区间Aa, Aa+1, ... , Ab.,第个数增加c, -10000 ≤ c ≤ 10000.
"Q a b" 表示查询区间Aa, Aa+1, ... , Ab之和
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
思路分析:
首先分析数据规模,如果有简单模拟算法,时间复杂度O(N*Q),肯定会超时;
看题目需求,每次输入都是对区间进行增加或者查询等操作,所以也必须用区间操作算法来解决;
此文以线段树来解决,树状数组也很适合解决此类问题。事实证明能用树状数组解决的都可以用线段树来解决,反之则不行;
线段树维护区间信息,而树状数组维护前缀和信息;
如上图P1.1,对于增加操作如(a=0, b=2, c=1),需要对[0,2]区间每个点进行增加,但如果递归所有子树去增加操作,就退化成了线性;
所以遍历到[0,2]区间时,只对该点操作,并记录以该点以根的子树,应该增加的值,先记录,下次遍历到子树再执行操作
如上图P1.2,对[0,2]执行了操作,再子树还没有增加,如果下次查询子树则数据错误,所以每次在执行增加或者查询操作的时候,就把上次遗留下来的数据给补上,也就是所谓的Lazy思想。
每个树节点主要记录如下信息:
left: 区间左起点
right: 区间右终点
total: 区间和
childrenExtra: 子树还应该额外增加的值
lson, rson: 左右子树
注:因题目数据量大,输入输出用scanf, printf,如果用cin, cout则会超时。
C++源码如下:
github: https://github.com/Kyle-Wilson1/Poj/tree/master/P3468
#include <iostream>
#include <fstream>
#include <vector> using namespace std; struct SegmentTree {
long long left, right, total, childrenExtra;
SegmentTree *lson, *rson;
}; long long regionLength(SegmentTree *tree) {
return tree->right - tree->left + 1;
} void updateChildren(SegmentTree *root) {
if (root->lson != NULL && root->rson != NULL && root->childrenExtra != 0) {
root->lson->total += root->childrenExtra * regionLength(root->lson);
root->lson->childrenExtra += root->childrenExtra;
root->rson->total += root->childrenExtra * regionLength(root->rson);
root->rson->childrenExtra += root->childrenExtra;
root->childrenExtra = 0;
}
} SegmentTree *buildTree(vector<long long> &num, long long l, long long r) {
if (l > r) {
return NULL;
}
if (l == r) {
SegmentTree *root = new SegmentTree;
root->left = l;
root->right = r;
root->total = num[l];
root->childrenExtra = 0;
root->lson = NULL;
root->rson = NULL;
return root;
}
SegmentTree *root = new SegmentTree;
root->left = l;
root->right = r;
root->childrenExtra = 0;
long long mid = (l + r) >> 1;
root->lson = buildTree(num, l, mid);
root->rson = buildTree(num, mid + 1, r);
root->total = root->lson->total + root->rson->total;
return root;
} void addRegion(SegmentTree *root, long long regionLeft, long long regionRight, long long addNum) {
if (root == NULL || root->right < regionLeft || root->left > regionRight) {
return;
}
if (root->left >= regionLeft && root->right <= regionRight) {
root->total += addNum * regionLength(root);
if (root->left < root->right)root->childrenExtra += addNum;
return;
}
updateChildren(root);
addRegion(root->lson, regionLeft, regionRight, addNum);
addRegion(root->rson, regionLeft, regionRight, addNum);
root->total = root->lson->total + root->rson->total;
} void queryRegion(SegmentTree *root, long long regionLeft, long long regionRight, long long &sum) {
if (root == NULL || root->right < regionLeft || root->left > regionRight) {
return;
}
if (root->left >= regionLeft && root->right <= regionRight) {
sum += root->total;
return;
}
updateChildren(root);
queryRegion(root->lson, regionLeft, regionRight, sum);
queryRegion(root->rson, regionLeft, regionRight, sum);
}
int main() {
ifstream fin("a.in");
ofstream fout("a.out"); long long n, q, i, a, b, c;
char type[2];
SegmentTree *root; // scanf("%lld%lld", &n, &q);
fin >> n >> q; vector<long long> num(n, 0); for (i = 0; i < n; i++) {
// scanf("%lld", &num[i]);
fin >> num[i];
} n--;
root = buildTree(num, 0, n); for (i = 0; i < q; i++) {
// scanf("%s", type);
fin >> type;
if (type[0] == 'C') {
// scanf("%lld%lld%lld", &a, &b, &c);
fin >> a >> b >> c;
addRegion(root, a - 1, b - 1, c);
} else {
// scanf("%lld%lld", &a, &b);
fin >> a >> b;
long long sum = 0;
queryRegion(root, a - 1, b - 1, sum);
// printf("%lld\n", sum);
fout << sum << endl;
}
} return 0;
}
Poj-P3468题解【线段树】的更多相关文章
- POJ.2299 Ultra-QuickSort (线段树 单点更新 区间求和 逆序对 离散化)
POJ.2299 Ultra-QuickSort (线段树 单点更新 区间求和 逆序对 离散化) 题意分析 前置技能 线段树求逆序对 离散化 线段树求逆序对已经说过了,具体方法请看这里 离散化 有些数 ...
- POJ2182题解——线段树
POJ2182题解——线段树 2019-12-20 by juruoOIer 1.线段树简介(来源:百度百科) 线段树是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间对应线 ...
- Buy Tickets POJ - 2828 思维+线段树
Buy Tickets POJ - 2828 思维+线段树 题意 是说有n个人买票,但是呢这n个人都会去插队,问最后的队列是什么情况.插队的输入是两个数,第一个是前面有多少人,第二个是这个人的编号,最 ...
- POJ 3468 A Simple Problem with Integers(详细题解) 线段树
这是个线段树题目,做之前必须要有些线段树基础才行不然你是很难理解的. 此题的难点就是在于你加的数要怎么加,加入你一直加到叶子节点的话,复杂度势必会很高的 具体思路 在增加时,如果要加的区间正好覆盖一个 ...
- poj City Horizon (线段树+二分离散)
http://poj.org/problem?id=3277 City Horizon Time Limit: 2000MS Memory Limit: 65536K Total Submissi ...
- poj 3667 Hotel (线段树)
http://poj.org/problem?id=3667 Hotel Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 94 ...
- poj 3225 【线段树】
poj 3225 这题是用线段树解决区间问题,看了两天多,算是理解一点了. Description LogLoader, Inc. is a company specialized in provid ...
- POJ 2352 Stars 线段树
题目链接 题意:在一个二维平面上有n个星星,每个星星的等级为x,x为该星星左方和下方所包含的星星的数量(包含正左和正下的),输出每个等级各有多少星星,星星坐标按照y序递增给出,y值相同按照x递增给出. ...
- poj 3264(线段树)
http://poj.org/problem?id=3264 初学线段可以做的水题,也是线段树的基础运用.也是我的第一个线段树的题. 题意:在区间范围内的最大值减去最小值 思路:线段树记录下每个区间内 ...
- POJ 3667 Hotel(线段树 区间合并)
Hotel 转载自:http://www.cnblogs.com/scau20110726/archive/2013/05/07/3065418.html [题目链接]Hotel [题目类型]线段树 ...
随机推荐
- 简易CLI
使用C语言实现一个简易的CLI,命令通过模式进行划分,实现效果如下: 代码较为简单,主要是为了方便进行移植,这里就不进行详细的说明了. 代码路径:https://github.com/zhengcix ...
- react第十单元(children的深入用法-React.Children对象上的方法)
第十单元(children的深入用法-React.Children对象上的方法) #课程目标 理解什么是children 掌握React.Children对象上的方法 #知识点 什么是children ...
- 移动端 CSS3动画属性
一.transform 转换属性 #1. translate位移 transform : translate(50px,100px); //把元素水平移动 50 像素,垂直移动 100 像素 tran ...
- Windows下python+allure的下载、安装、配置与使用
下载安装allure 1.Windows和mac均可选择从官网下载,下载地址: https://repo.maven.apache.org/maven2/io/qameta/allure/allure ...
- 一个小技巧助您减少if语句的状态判断
作者:依乐祝 首发地址:https://www.cnblogs.com/yilezhu/p/14174990.html 在进行项目的开发的过程中, if 语句是少不了的,但我们始终要有一颗消灭 if ...
- 微博爬虫,python微博用户主页小姐姐图片内容采集爬虫
python爬虫,微博爬虫,需要知晓微博用户id号,能够通过抓取微博用户主页内容来获取用户发表的内容,时间,点赞数,转发数等数据,当然以上都是本渣渣结合网上代码抄抄改改获取的! 要抓取的微博地址:ht ...
- python函数收集不确定数量的值
python写函数的时候,有时候会不确定到底传入多少值. 首先是,*args,单星号参数收集参数: 1 #!usr/bin/python 2 #-*-coding:utf-8-*- 3 4 #定义一个 ...
- MethodHandleVS反射
Method Handle与反射 如无特殊说明,本文所有代码均基于JDK1.8.0_221 Method Handle入门 反射我们都知道,为我们提供了运行时对类的成员方法访问的手段,极大地提高了Ja ...
- xxfpmW 的诞生过程
最近因为在win 服务器搭建php服务,发现php-cgi.exe 很容易崩溃,看cpu和硬盘都没有暴涨,也不知道啥原因,网上查发现有一款xxfpm 小应用可以解决这个问题,但这个应用是2011年开发 ...
- 【kinetic】操作系统探索总结(八)键盘控制
如果尝试过前面的例子,有没有感觉每次让机器人移动还要在终端里输入指令,这也太麻烦了,有没有办法通过键盘来控制机器人的移动呢?答案室当然的了.我研究了其他几个机器人键盘控制的代码,还是有所收获的,最后移 ...