uestc 360(区间合并)
题意:有一个长度为n的序列。然后有两种操作,Q a b是输出区间a b内最长上升子序列的长度。A a b c是把区间a b内全部数字加上c。
题解:用线段树维护区间的最长上升子序列长度,那么一个区间的最长上升子序列有三种可能,从左端点开头的一段,或者中间的一段,或者中间到右端点一段。所以线段树应该要维护liml和limr表示当前区间左端点向右延伸上升长度和右端点向左延伸的上升长度,然后在求当前区间的最长上升子序列长度时,是左子区间的和右子区间的上升子序列长度,以及当前区间的liml和limr的较大值。
假设左子区间右端点小于右子区间左端点。那么还能够是左子区间limr和右子区间的liml的合并。
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int N = 100005;
struct Tree {
int addv, liml, limr;
int len, lv, rv;
}tree[N << 2];
int n, q, a[N];
char op[5];
void pushup(int k, int left, int right) {
int lenn = right - left + 1;
tree[k].lv = tree[k * 2].lv;
tree[k].rv = tree[k * 2 + 1].rv;
tree[k].liml = tree[k * 2].liml;
tree[k].limr = tree[k * 2 + 1].limr;
tree[k].len = max(tree[k * 2].len, tree[k * 2 + 1].len);
if (tree[k * 2].rv < tree[k * 2 + 1].lv) {
if (tree[k * 2].liml == lenn - (lenn / 2))
tree[k].liml += tree[k * 2 + 1].liml;
if (tree[k * 2 + 1].limr == lenn / 2)
tree[k].limr += tree[k * 2].limr;
tree[k].len = max(tree[k].len, tree[k * 2].limr + tree[k * 2 + 1].liml);
}
tree[k].len = max(tree[k].len, max(tree[k].liml, tree[k].limr));
}
void pushdown(int k) {
if (tree[k].addv) {
tree[k * 2].addv += tree[k].addv;
tree[k * 2 + 1].addv += tree[k].addv;
tree[k * 2].lv += tree[k].addv;
tree[k * 2 + 1].lv += tree[k].addv;
tree[k * 2].rv += tree[k].addv;
tree[k * 2 + 1].rv += tree[k].addv;
tree[k].addv = 0;
}
}
void build(int k, int left, int right) {
tree[k].addv = 0;
if (left == right) {
tree[k].lv = tree[k].rv = a[left];
tree[k].liml = tree[k].limr = tree[k].len = 1;
return;
}
int mid = (left + right) / 2;
build(k * 2, left, mid);
build(k * 2 + 1, mid + 1, right);
pushup(k, left, right);
}
void modify(int k, int left, int right, int l, int r, int x) {
if (l <= left && right <= r) {
tree[k].addv += x;
tree[k].lv += x;
tree[k].rv += x;
return;
}
pushdown(k);
int mid = (left + right) / 2;
if (l <= mid)
modify(k * 2, left, mid, l, r, x);
if (r > mid)
modify(k * 2 + 1, mid + 1, right, l, r, x);
pushup(k, left, right);
}
int query(int k, int left, int right, int l, int r) {
if (l <= left && right <= r)
return tree[k].len;
pushdown(k);
int mid = (left + right) / 2;
if (mid >= r)
return query(k * 2, left, mid, l, r);
if (mid < l)
return query(k * 2 + 1, mid + 1, right, l, r);
int res = max(query(k * 2, left, mid, l, mid), query(k * 2 + 1, mid + 1, right, mid + 1, r));
if (tree[k * 2].rv < tree[k * 2 + 1].lv) {
int temp1 = min(mid - l + 1, tree[k * 2].limr);
int temp2 = min(r - mid, tree[k * 2 + 1].liml);
res = max(res, temp1 + temp2);
}
return res;
}
int main() {
int t, cas = 1;
scanf("%d", &t);
while (t--) {
scanf("%d%d", &n, &q);
for (int i = 1; i <= n; i++)
scanf("%d", &a[i]);
build(1, 1, n);
printf("Case #%d:\n", cas++);
int l, r, x;
while (q--) {
scanf("%s", op);
if (op[0] == 'q') {
scanf("%d%d", &l, &r);
printf("%d\n", query(1, 1, n, l, r));
}
else {
scanf("%d%d%d", &l, &r, &x);
modify(1, 1, n, l, r, x);
}
}
}
return 0;
}
uestc 360(区间合并)的更多相关文章
- UESTC 360(1425) another LCIS
这道题是CD老OJ上面的一道题,现在在新OJ上的题号是360,开始在VJ上做的提交一直RE(囧).后来才知道OJ移位了. 这道题是一个简单的成段更新+区间合并的线段树的题,1A还让我小激动了一下 这道 ...
- POJ 3667 Hotel(线段树 区间合并)
Hotel 转载自:http://www.cnblogs.com/scau20110726/archive/2013/05/07/3065418.html [题目链接]Hotel [题目类型]线段树 ...
- HDU 3911 线段树区间合并、异或取反操作
题目:http://acm.hdu.edu.cn/showproblem.php?pid=3911 线段树区间合并的题目,解释一下代码中声明数组的作用: m1是区间内连续1的最长长度,m0是区间内连续 ...
- HDU 1540 Tunnel Warfare 平衡树 / 线段树:单点更新,区间合并
Tunnel Warfare Time Limit: 4000/2000 MS (Java/Others) Memory Lim ...
- HDU 3911 Black And White(线段树区间合并+lazy操作)
开始以为是水题,结果...... 给你一些只有两种颜色的石头,0为白色,1为黑色. 然后两个操作: 1 l r 将[ l , r ]内的颜色取反 0 l r 计算[ l , r ]内最长连续黑色石头的 ...
- POJ 2750 Potted Flower (线段树区间合并)
开始懵逼找不到解法,看了网上大牛们的题解才发现是区间合并... 给你n个数形成一个数列环,然后每次进行一个点的修改,并输出这个数列的最大区间和(注意是环,并且区间最大只有n-1个数) 其实只需要维护 ...
- ACM: Hotel 解题报告 - 线段树-区间合并
Hotel Time Limit:3000MS Memory Limit:65536KB 64bit IO Format:%lld & %llu Description The ...
- HDU 1540 Tunnel Warfare(线段树+区间合并)
http://acm.hdu.edu.cn/showproblem.php?pid=1540 题目大意:抗日战争期间进行地道战,存在n个村庄用地道连接,输入D表示破坏某个村庄(摧毁与其相连的地道, 包 ...
- POJ 2892 Tunnel Warfare(线段树单点更新区间合并)
Tunnel Warfare Time Limit: 1000MS Memory Limit: 131072K Total Submissions: 7876 Accepted: 3259 D ...
随机推荐
- ubuntu下安装低级版本gcc/g++ 并随意切换
来自:http://blog.sina.com.cn/s/blog_6cee149d010129bl.html 发现Android的版本中编译Host的程序经常因为本机的Gcc版本过高,需要这样那样的 ...
- "characterEncoding" must end with the ';' delimiter.
17/04/20 17:27:10 FATAL conf.Configuration: error parsing conf file:/usr/local/apache-hive-1.2.2-bin ...
- 洛谷P4645 [COCI2006-2007 Contest#7] BICIKLI [Tarjan,拓扑排序]
题目传送门 BICIKLI 题意翻译 给定一个有向图,n个点,m条边.请问,1号点到2号点有多少条路径?如果有无限多条,输出inf,如果有限,输出答案模10^9的余数. 两点之间可能有重边,需要看成是 ...
- Mybatis 源码分析之事物管理
Mybatis 提供了事物的顶层接口: public interface Transaction { /** * Retrieve inner database connection * @retur ...
- Struts2的概念
Struts2的概念 Struts2是一个基于MVC设计模式的Web应用框架,它本质上相当于一个servlet,在MVC设计模式中,Struts2作为控制器(Controller)来建立模型与视图的数 ...
- 二分搜索之C++实现
二分搜索之C++实现 一.源代码:BinarySearch.cpp #include<iostream> using namespace std; /*定义输出一维数组的函数*/ void ...
- 【堆优化Dijkstra】BZOJ4152- [AMPPZ2014]The Captain
[题目大意] 给定平面上的n个点,定义(x1,y1)到(x2,y2)的费用为min(|x1-x2|,|y1-y2|),求从1号点走到n号点的最小费用. [思路] 按照某维坐标排序,相邻两个点在这一维度 ...
- hdu 3879 方案选择
每日一水--- #include <cstdio> #include <cstring> #include <vector> #define oo 0x3f3f3f ...
- Express浅析
一.Express是什么? 首先Express是一个Node.js的Web框架,它的API使用各种HTTP实用程序方法和中间件,快速方便地创建强大的API,性能方面Express提供精简的基本web应 ...
- 通过IP获取所在城市
<script type="text/javascript"> var map = new BMap.Map("allmap"); var poin ...