南邮OJ 2027 操作序列
来源:互联网 发布:java培训育知同创 编辑:程序博客网 时间:2024/05/29 19:56
链接:http://acm.njupt.edu.cn/acmhome/problemdetail.do?&method=showdetail&id=2027
题目:
操作序列
总提交:146 测试通过:26
描述
给出一初始序列a1, a2,...,an,下面有m个操作(x, l, r) : 对于a[l], a[l+1],...,a[r]都加上x.
输出m个操作结束后的序列.
输入
第一行两个整数n,m(0 <= n,m <= 100000),n表序列{A}的长度, m表操作的个数。
第二行有n 个整数ai(-10000 <= ai <= 10000)。
下面m行,每一行表示一个操作,一个操作表示为3个整数x, l, r(1 <= l <= r <= n, |x|<=1000)。
输出
输出结果序列。数据已改正,行末没有空格!
样例输入
5 3
1 2 3 -4 5
2 1 1
-3 3 5
0 1 5
样例输出
3 2 0 -7 2
提示
null
解题思路:
解法1:
这道题目,主要涉及的就是一个区间更新的问题。将数组a的[L,R]区间内的每个元素加x。我们可以这样来解决这个问题,将区间[L,R]分解为[1,R]和[1,L-1],这样原问题就可以转化为,将[1,R]的每个元素加x,再将[1,L-1]的每个元素加(-x),我们可以用一个数组sum来记录这些更新。sum[i] = x,表示对数组a从第1个元素到第i个元素,每个元素加上x。这样的话最后我们对a的更新就变成了这样:a[i] = a[i] + sum[i] + sum[i+1] + sum[i+2] + ... + sum[n];我们可以优化一下,将sum数组进行处理,使其成为前缀和数组(sum[i] = b[1] + b[2] + ... + b[i] )。这样对a的更新就变成了:a[i] = a[i] + sum[n] - sum[i-1];
解法2:
线段树区间更新。
解法1代码:
#include <iostream>#include <cstring>using namespace std;const int MAXN = 100010;int a[MAXN], sum[MAXN];//sum[i] = x数组,表示a[i]数组从1到i每个元素加x int main(){std::ios::sync_with_stdio(false);memset(a, 0, sizeof(a));memset(sum, 0, sizeof(sum));int n, m;cin >> n >> m;for(int i = 1; i <= n; i++)cin >> a[i];while(m--){int l, r, x;cin >> x >> l >> r;sum[r]+= x;sum[l-1]-= x;}for(int i = 1; i <= n; i++)sum[i] += sum[i-1];for(int i = 1; i <= n; i++)a[i] += (sum[n] - sum[i-1]);for(int i = 1; i <= n; i++){if(i - 1) cout << " ";cout << a[i];}cout << endl;return 0;}
解法2代码:
#include <iostream>#include <cstring>#include <cstdio>using namespace std;const int MAXN = 100010;struct Tree{int left, right, x;};Tree tree[4 * MAXN];int first = 1;void build_tree(int l, int r, int i){tree[i].left = l;tree[i].right = r;tree[i].x = 0;if(l == r){return ;}int mid = (l + r) >> 1;build_tree(l, mid, i + i);build_tree(mid + 1, r, i + i + 1);}void update(int l, int r, int i, int x){if(l == tree[i].left && r == tree[i].right){tree[i].x += x;return;}int mid = (tree[i].left + tree[i].right) >> 1;if(r <= mid){update(l, r, i + i, x);}else{if(l > mid){update(l, r, i + i + 1, x);}else{update(l, mid, i + i, x);update(mid + 1, r, i + i + 1, x);}}}void print(int l, int r, int i, int num){if(l == r){if(!first) cout << " ";else first = 0;cout << tree[i].x + num;return ;}int mid = (l + r) >> 1;print(l, mid, i + i, tree[i].x + num);print(mid + 1, r, i + i + 1, tree[i].x + num);}int main(){std::ios::sync_with_stdio(false);int n, m;cin >> n >> m;tree[1].left = 1, tree[1].right = n, tree[1].x = 0;build_tree(1, n, 1);for(int i = 1; i <= n; i++){int x;cin >> x;update(i, i, 1, x);}while(m--){int x, l, r;cin >> x >> l >> r;update(l, r, 1, x);}print(1, n, 1, 0);return 0;}
- 南邮OJ 2027 操作序列
- 南邮 OJ 2027 操作序列
- 南邮 OJ 1043 合法序列
- 南邮 OJ 1618 合法序列
- 【HPU OJ 1310 】序列的区间操作 【思维】
- 南邮 OJ 2024 入栈序列和出栈序列
- 华为OJ:DNA序列
- 【华为OJ】DNA序列
- 华为oj DNA序列
- 华为oj DNA序列
- OJ---DNA序列*
- 【华为OJ】DNA序列
- 华为OJ:DNA序列
- DNA序列(OJ)
- oj 序列判断
- 南邮 OJ 1047 图的深度优先遍历序列
- 南邮 OJ 1048 图的宽度优先遍历序列
- 南邮 OJ 1004 线性表操作
- JQuery取RadioButtonList 或者 CheckBox的一些值的方法 [转]
- 利用maven将项目依赖的jar提取到指定文件夹
- hadoop中hive和hbase的整合应用
- vim使用Vundle一键配置完美php IDE开发环境
- Java 8 彻底改变数据库访问
- 南邮OJ 2027 操作序列
- Cocos2d-x 框架概要说明
- 打印当前日期和时间,记录微秒级时间日志的C程序
- Linux服务器网络开发模型
- C++ 析构函数 构造函数 虚析构函数
- 让你提前认识软件开发(7):变量及函数的命名
- 以后这里就是我大脑的小硬盘啦
- 控制台中输出宽字符
- 动态设置表结构方案(一)