uestc 94(区间更新)
题意:有一个字符串全部由’(‘和’)’组成。然后有三种操作,query a b输出区间[a,b]字符串的括号序列是否合法。reverse a b把区间[a,b]字符串里全部’(‘替换成’)’,而且把全部’)’替换为’(‘,set a b c,把区间[a,b]的全部字符替换为c。
题解:明显是线段树,为了能够让线段树维护,推断一个字符串是否为合法括号,能够把全部的’(‘替换为-1,’)’替换为1,那么假设这个字符串合法,整个字符串的和应该是0且前缀最大和不能超过0。所以能够在线段树节点加入maxx和sum这两个值维护区间内的前缀最大和和总和。
然后reverse操作时。为了方便计算maxx,节点应该还要维护minn前缀最小和,这样maxx = -minn能够直接计算。还有区间合并时最大和是选左子区间最大和与左子区间总和加上右子区间的最大和中较大的那个。最小和求法同理。
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int N = 100005;
struct Tree {
int sum, maxx, minn;
int setv, rev;
void f(int val, int left, int right) {
if (!val) {
sum = -sum;
int temp = maxx;
maxx = -minn;
minn = -temp;
rev ^= 1;
}
else {
minn = min(val, val * (right - left + 1));
maxx = max(val, val * (right - left + 1));
sum = val * (right - left + 1);
setv = val;
rev = 0;
}
}
}tree[N << 2];
int n, q, a[N];
char s[N], op[10], c[5];
void pushdown(int k, int left, int right) {
int mid = (left + right) / 2;
if (tree[k].setv) {
tree[k * 2].f(tree[k].setv, left, mid);
tree[k * 2 + 1].f(tree[k].setv, mid + 1, right);
tree[k].setv = 0;
}
if (tree[k].rev) {
tree[k * 2].f(0, left, mid);
tree[k * 2 + 1].f(0, mid + 1, right);
tree[k].rev = 0;
}
}
void pushup(int k) {
tree[k].sum = tree[k * 2].sum + tree[k * 2 + 1].sum;
tree[k].maxx = max(tree[k * 2].maxx, tree[k * 2].sum + tree[k * 2 + 1].maxx);
tree[k].minn = min(tree[k * 2].minn, tree[k * 2].sum + tree[k * 2 + 1].minn);
}
void build(int k, int left, int right) {
tree[k].setv = tree[k].rev = 0;
if (left == right) {
tree[k].sum = tree[k].maxx = tree[k].minn = a[left];
return;
}
int mid = (left + right) / 2;
build(k * 2, left, mid);
build(k * 2 + 1, mid + 1, right);
pushup(k);
}
void modify(int k, int left, int right, int l, int r, int v) {
if (l <= left && right <= r) {
tree[k].f(v, left, right);
return;
}
pushdown(k, left, right);
int mid = (left + right) / 2;
if (l <= mid)
modify(k * 2, left, mid, l, r, v);
if (r > mid)
modify(k * 2 + 1, mid + 1, right, l, r, v);
pushup(k);
}
void query(int k, int left, int right, int l, int r, int &sum, int &maxx) {
if (l <= left && right <= r) {
maxx = tree[k].maxx;
sum = tree[k].sum;
return;
}
pushdown(k, left, right);
int mid = (left + right) / 2;
if (r <= mid)
query(k * 2, left, mid, l, r, sum, maxx);
else if (l > mid)
query(k * 2 + 1, mid + 1, right, l, r, sum, maxx);
else {
int sum1, maxx1, sum2, maxx2;
query(k * 2, left, mid, l, mid, sum1, maxx1);
query(k * 2 + 1, mid + 1, right, mid + 1, r, sum2, maxx2);
sum = sum1 + sum2;
maxx = max(maxx1, sum1 + maxx2);
}
pushup(k);
}
int main() {
int t, cas = 1;
scanf("%d", &t);
while (t--) {
scanf("%d%s", &n, s + 1);
for (int i = 1; i <= n; i++)
if (s[i] == '(')
a[i] = -1;
else
a[i] = 1;
build(1, 1, n);
printf("Case %d:\n", cas++);
scanf("%d", &q);
int l, r, sum, maxx;
while (q--) {
scanf("%s", op);
if (op[0] == 'q') {
scanf("%d%d", &l, &r);
query(1, 1, n, l + 1, r + 1, sum, maxx);
if (!sum && maxx <= 0)
printf("YES\n");
else
printf("NO\n");
}
else if (op[0] == 's') {
scanf("%d%d%s", &l, &r, c);
if (c[0] == '(')
modify(1, 1, n, l + 1, r + 1, -1);
else
modify(1, 1, n, l + 1, r + 1, 1);
}
else {
scanf("%d%d", &l, &r);
modify(1, 1, n, l + 1, r + 1, 0);
}
}
printf("\n");
}
return 0;
}
uestc 94(区间更新)的更多相关文章
- 2015 UESTC 数据结构专题A题 秋实大哥与小朋友 线段树 区间更新,单点查询,离散化
秋实大哥与小朋友 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.uestc.edu.cn/#/contest/show/59 Desc ...
- Codeforces Round #419 (Div. 2) A B C 暴力 区间更新技巧 +前缀和 模拟
A. Karen and Morning time limit per test 2 seconds memory limit per test 512 megabytes input standar ...
- CDOJ 1059 秋实大哥与小朋友 STL(set)+离散化+BIT区间更新单点查询
链接: A - 秋实大哥与小朋友 Time Limit:1000MS Memory Limit:65535KB 64bit IO Format:%lld & %llu Subm ...
- HDU 1556 Color the ball(线段树区间更新)
Color the ball 我真的该认真的复习一下以前没懂的知识了,今天看了一下线段树,以前只会用模板,现在看懂了之后,发现还有这么多巧妙的地方,好厉害啊 所以就应该尽量搞懂 弄明白每个知识点 [题 ...
- hihoCoder 1080 : 更为复杂的买卖房屋姿势 线段树区间更新
#1080 : 更为复杂的买卖房屋姿势 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho都是游戏迷,“模拟都市”是他们非常喜欢的一个游戏,在这个游戏里面他们 ...
- hdu 3397 Sequence operation(线段树:区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3397 题意:给你一个长度为n的0,1序列,支持下列五种操作, 操作0(0 a b):将a到b这个区间的 ...
- 【HDU 4614】Vases and Flowers(线段树区间更新懒惰标记)
题目0到n-1的花瓶,操作1在下标a开始插b朵花,输出始末下标.操作2清空[a,b]的花瓶,求清除的花的数量.线段树懒惰标记来更新区间.操作1,先查询0到a-1有num个空瓶子,然后用线段树的性质,或 ...
- HDU 1698 Just a Hook(线段树/区间更新)
题目链接: 传送门 Minimum Inversion Number Time Limit: 1000MS Memory Limit: 32768 K Description In the g ...
- HDU 5023 A Corrupt Mayor's Performance Art(线段树区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5023 解题报告:一面墙长度为n,有N个单元,每个单元编号从1到n,墙的初始的颜色是2,一共有30种颜色 ...
随机推荐
- CTSC2012 熟悉的文章 广义后缀自动机_单调队列
没啥难的,主要是单调队列忘了咋求了QAQ... Code: #include <cstdio> #include <algorithm> #include <cstrin ...
- [USACO10FEB]吃巧克力Chocolate Eating
题目:洛谷P2985. 题目大意:有n块巧克力要吃d天,并且只能按顺序吃.一块巧克力有一个开心值,吃了就能增加开心值.一个人初始开心值为0,且每天早上开心值变为原来的一半.问如何吃巧克力才能使开心值最 ...
- ElementUi rules表单验证
ElementUi 表单验证 工作中常用到的JS验证 可以在pattern中书写正则,并且配合elementUI进行表单验证. pattern 属性规定用于验证输入字段的模式.模式指的是正则表达式. ...
- MD5 加密原理(转)
MD5的全称是Message-Digest Algorithm 5(信息-摘要算法),在90年代初由MIT Laboratory for Computer Science和RSA Data Secur ...
- 洛谷 P2665 [USACO08FEB]连线游戏Game of Lines
P2665 [USACO08FEB]连线游戏Game of Lines 题目背景 Farmer John最近发明了一个游戏,来考验自命不凡的贝茜. 题目描述 Farmer John has chall ...
- Qt之字典划词
简述 相信大家都用过词典吧!因为英语不太好...O(∩_∩)O~,所以经常进行划词翻译! 简述 实现 效果 源码 更多参考 实现 原理:鼠标移至某单词之上,获取鼠标位置,然后在对应位置进行取词,翻译! ...
- MapReduce中combine、partition、shuffle的作用是什么
http://www.aboutyun.com/thread-8927-1-1.html Mapreduce在hadoop中是一个比較难以的概念.以下须要用心看,然后自己就能总结出来了. 概括: co ...
- CorePlot学习六---点击scatterPlot中的symbol点时弹出对应的凝视
因为项目须要用到用户点击 symbol时,弹出对应的具体信息,发现国内解说的比較少,经过一番搜索验证最终解决,先看效果图: 详细须要改动的代码例如以下: 首先要引用托付方法:CPTScatterPlo ...
- nj03---阻塞和线程
Node.js最大的特性就是"异步式I/O"与事件紧密结合的编程模式.这种模式与传统的同步式IO线性的编程思路有很大的不同,因为控制流很大程度上要靠"事件"和& ...
- Android Jni 调用
Chap1:JNI完全手册... 3 Chap2:JNI-百度百科... 11 Chap 3:javah命令帮助信息... 16 Chap 4:用javah产生一个.h文件... 17 Chap5:j ...