题目大意:给定一个合法的括号序列(只包含'(',')'),有q次操作,对每次操作改变一个位置的括号,求最左端的位置,使得改变这个位置上的括号以后,新序列合法(完全配对)。

思路:对于合法的括号序列,如果把括号序列一次进行栈操作,把'('进栈,')'则把最近的'('出栈,令a[i]表示到i位置后栈里面的左括号个数,也就是i位置的左括号数目和右括号数目的差。则原序列对应新的数组a。原序列合法 等价于 对于1<=i<=n,a[i]>=0 恒成立。那么对于把位置pos上的括号改变一下,如果是'(' - ')',那么a[pos~n]会都减去2,如果是')'->'(',那么a[pos~n]会都加上2。分类讨论后,答案不难得出。对于'('->')'答案就是从左至右找第一个右括号;对于')'->'(',答案就是找一个最左端的位置pos0,使得a[pos0~pos-1]都大于等于2。对于找第一个右括号,可以转化为用新的数组b[i] = a[i] - i,找第一个小于0的位置,进而转化为找最大的前缀区间使得这个区间上的b的最小值等于0。对于找最左端的位置,也可以转化为区间最值来做,详见代码:

 #include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <map>
#include <queue>
#include <cmath>
#include <vector>
#include <ctime>
#define mem0(a) memset(a, 0, sizeof(a))
#define lson l, m, rt << 1
#define rson m + 1, r, rt << 1 | 1
#define define_m int m = (l + r) >> 1
#define LL long long
#define Rep(a, b) for(int a = 0; a < b; a++)
#define lowbit(x) ((x) & (-(x)))
const int dx[] = {, , -, };
const int dy[] = {, -, , };
const int INF = 1e9 + ;
const int maxn = ;
const double eps = 1e-;
typedef double db;
using namespace std;
struct SegTree {
struct Node {
int minv, add;
} tree[maxn << ];
void PushUp(int rt) {
tree[rt].minv = min(tree[rt << ].minv, tree[rt << | ].minv);
}
void PushDown(int rt) {
int add = tree[rt].add;
if (add) {
tree[rt << ].minv += add;
tree[rt << ].add += add;
tree[rt << | ].minv += add;
tree[rt << | ].add += add;
tree[rt].add = ;
}
}
void Build(int a[], int l, int r, int rt) {
tree[rt].add = ;
if (l == r) {
tree[rt].minv = a[l];
return ;
}
define_m;
Build(a, lson);
Build(a, rson);
PushUp(rt);
}
void Update(int L, int R, int x, int l, int r, int rt) {
if (L <= l && r <= R) {
tree[rt].minv += x;
tree[rt].add += x;
return;
}
define_m;
PushDown(rt);
if (L <= m) Update(L, R, x, lson);
if (R > m) Update(L, R, x, rson);
PushUp(rt);
}
int Query1(int l, int r, int rt) {
if (l == r) return tree[rt].minv == ;
define_m;
PushDown(rt);
if (tree[rt << ].minv < ) return Query1(lson);
int len = r - l + ;
return len - (len >> ) + Query1(rson);
}
int Query2(int l, int r, int rt) {
if (l == r) return tree[rt].minv == ;
define_m;
PushDown(rt);
if (tree[rt << | ].minv < ) return Query2(rson);
int len = r - l + ;
return (len >> ) + Query2(lson);
}
};
int n, q;
char s[];
int a[], b[], c[];
SegTree G, H;
void Init() {
for (int i = ; i < n; i++) {
c[i + ] = c[i];
if (s[i] == '(') c[i + ]++;
else c[i + ]--;
}
for (int i = ; i <= n; i++) {
a[i] = c[i] - i;
b[i] = c[i];
}
}
int main() {
//freopen("in.txt", "r", stdin);
cin >> n >> q;
scanf("%s", s);
Init();
G.Build(a, , n, );
H.Build(b, , n, );
for (int i = ; i < q; i++) {
int pos, tmp;
scanf("%d", &pos);
if (s[pos - ] == '(') {
s[pos - ] = ')';
G.Update(pos, n, -, , n, );
H.Update(pos, n, -, , n, );
printf("%d\n", tmp = G.Query1(, n, ) + );
G.Update(tmp, n, , , n, );
H.Update(tmp, n, , , n, );
s[tmp - ] = '(';
}
else {
s[pos - ] = '(';
H.Update(pos, n, , , n, );
G.Update(pos, n, , , n, );
printf("%d\n", tmp = n - H.Query2(, n, ) + );
H.Update(tmp, n, -, , n, );
G.Update(tmp, n, -, , n, );
s[tmp - ] = ')';
}
}
return ;
}

【csu oj 1542】线段树的更多相关文章

  1. hdu 1542 线段树扫描(面积)

    Atlantis Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Su ...

  2. luogu P3285 [SCOI2014]方伯伯的OJ splay 线段树

    LINK:方伯伯的OJ 一道稍有质量的线段树题目.不写LCT splay这辈子是不会单独写的 真的! 喜闻乐见的是 题目迷惑选手 \(op==1\) 查改用户在序列中的位置 题目压根没说位置啊 只有排 ...

  3. hdu 1542 线段树+扫描线 学习

    学习扫描线ing... 玄学的东西... 扫描线其实就是用一条假想的线去扫描一堆矩形,借以求出他们的面积或周长(这一篇是面积,下一篇是周长) 扫描线求面积的主要思想就是对一个二维的矩形的某一维上建立一 ...

  4. 玲珑oj 1117 线段树+离线+离散化,laz大法

    1117 - RE:从零开始的异世界生活 Time Limit:1s Memory Limit:256MByte Submissions:438Solved:68 DESCRIPTION 486到了异 ...

  5. CSUOJ 1542 线段树解决括号反向问题

    题目大意: 根据初始给定的合法的小括号排序,每次进行一个操作,将第a位的括号反向,找到一个尽可能靠前的括号反向后是整个括号排列合法 数据量十分大,不断进行查询,要用线段树进行logn的复杂度的查询 首 ...

  6. hdu 1542 线段树之扫描线之面积并

    点击打开链接 题意:给你n个矩形,求它们的面积,反复的不反复计算 思路:用线段树的扫描线完毕.将X坐标离散化后,从下到上扫描矩形,进行各种处理,看代码凝视把 #include <stdio.h& ...

  7. Atlantis HDU - 1542 线段树+扫描线 求交叉图形面积

    //永远只考虑根节点的信息,说明在query时不会调用pushdown //所有操作均是成对出现,且先加后减 // #include <cstdio> #include <cstri ...

  8. HDU 1542 线段树离散化+扫描线 平面面积计算

    也是很久之前的题目,一直没做 做完之后觉得基本的离散化和扫描线还是不难的,由于本题要离散x点的坐标,最后要计算被覆盖的x轴上的长度,所以不能用普通的建树法,建树建到r-l==1的时候就停止,表示某段而 ...

  9. HDU 1542 线段树+扫描线+离散化

    Atlantis Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Su ...

随机推荐

  1. mybatis 批量删除

    mapper.xml: <update id="delete" parameterType="int"> delete from user_logi ...

  2. Oracle使用fy_recover_data恢复truncate删除的数据

    (一)truncate操作概述 在生产中,truncate是使用的多的命令,在使用不当的情况下,往往会造成表的数据全部丢失,恢复较为困难.对于truncate恢复,常见的有以下几种方法可以进行恢复: ...

  3. tensorflow--filter、strides

    最近还在看<TensorFlow 实战Google深度学习框架第二版>这本书,根据第六章里面对于卷基层和池化层的介绍可以发现,在执行 tf.nn.conv2d 和 tf.nn.max_po ...

  4. 8. input限制手机输入

    1. 只能输入数字: <input id="num" type="number" value="0" onkeyup="va ...

  5. redis:String字符串类型(三)

    字符串拼接(如果key不存在则创建):append name " applesnt" 获取字符串的长度:strlen name 127.0.0.1:6379> set nam ...

  6. 解决sublime打开文档,出现中文乱码问题

    sublime text 软件中出现中文乱码,大多是因为编码格式不支持,需要安装一个插件就可以解决中文乱码问题,推荐安装 ConvertToUtf8  安装步骤: 1.按“shift + ctrl + ...

  7. 小白必看,Python 各种下划线都是啥意思_、_xx、xx_、__xx、__xx__、_classname_

    我们在定义一些变量或者方法的时候,常常会用到下划线,在 Python 中,下划线可是很有用处的哟,比如变量,有些是一个下划线开头的(_xx),有些是两个下划线开头的(__xx),有些是在名称的结尾添加 ...

  8. Jenkins(2)- 更改插件源为国内源

    如果想从头学起Jenkins的话,可以看看这一系列的文章哦 https://www.cnblogs.com/poloyy/category/1645399.html jenkins插件清华大学镜像地址 ...

  9. tp5--相对路径和绝对路径

    首先,我们要先明白相对路径和绝对路径的理论: 绝对路径:是从盘符开始的路径,形如C:\windows\system32\cmd.exe相对路径:是从当前路径开始的路径,假如当前路径为C:\window ...

  10. python学习03字符串基本操作

    '''字符串可以用单引号,双引号,三引号表示 '''#1.读取str1='I am a student!'#每一个字符对应一个下标,可以利用下标的方式来读取字符串对应的值——索引print(str1[ ...