【csu oj 1542】线段树
题目大意:给定一个合法的括号序列(只包含'(',')'),有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】线段树的更多相关文章
- hdu 1542 线段树扫描(面积)
Atlantis Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Su ...
- luogu P3285 [SCOI2014]方伯伯的OJ splay 线段树
LINK:方伯伯的OJ 一道稍有质量的线段树题目.不写LCT splay这辈子是不会单独写的 真的! 喜闻乐见的是 题目迷惑选手 \(op==1\) 查改用户在序列中的位置 题目压根没说位置啊 只有排 ...
- hdu 1542 线段树+扫描线 学习
学习扫描线ing... 玄学的东西... 扫描线其实就是用一条假想的线去扫描一堆矩形,借以求出他们的面积或周长(这一篇是面积,下一篇是周长) 扫描线求面积的主要思想就是对一个二维的矩形的某一维上建立一 ...
- 玲珑oj 1117 线段树+离线+离散化,laz大法
1117 - RE:从零开始的异世界生活 Time Limit:1s Memory Limit:256MByte Submissions:438Solved:68 DESCRIPTION 486到了异 ...
- CSUOJ 1542 线段树解决括号反向问题
题目大意: 根据初始给定的合法的小括号排序,每次进行一个操作,将第a位的括号反向,找到一个尽可能靠前的括号反向后是整个括号排列合法 数据量十分大,不断进行查询,要用线段树进行logn的复杂度的查询 首 ...
- hdu 1542 线段树之扫描线之面积并
点击打开链接 题意:给你n个矩形,求它们的面积,反复的不反复计算 思路:用线段树的扫描线完毕.将X坐标离散化后,从下到上扫描矩形,进行各种处理,看代码凝视把 #include <stdio.h& ...
- Atlantis HDU - 1542 线段树+扫描线 求交叉图形面积
//永远只考虑根节点的信息,说明在query时不会调用pushdown //所有操作均是成对出现,且先加后减 // #include <cstdio> #include <cstri ...
- HDU 1542 线段树离散化+扫描线 平面面积计算
也是很久之前的题目,一直没做 做完之后觉得基本的离散化和扫描线还是不难的,由于本题要离散x点的坐标,最后要计算被覆盖的x轴上的长度,所以不能用普通的建树法,建树建到r-l==1的时候就停止,表示某段而 ...
- HDU 1542 线段树+扫描线+离散化
Atlantis Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Su ...
随机推荐
- python 利用numpy同时打乱列表的顺序,同时打乱数据和标签的顺序
可用于网络训练打乱训练数据个标签,不改变对应关系 方法一: np.random.shuffle (无返回值,直接打乱原列表) state = np.random.get_state() np.rand ...
- bm25算法和tfidf
- Caused by: java.lang.ClassCastException: class java.lang.Double cannot be cast to class org.apache.hadoop.io.WritableComparable
错误: Caused by: java.lang.ClassCastException: class java.lang.Double cannot be cast to class org.apac ...
- xpath进阶用法
一.简介 xpath作为对网页.对xml文件进行定位的工具,速度快,语法简洁明了,在网络爬虫解析内容的过程中起到很大的作用,除了xpath的基础用法之外xpath中还存在着非常之多的进阶用法,本文将对 ...
- MySql -- 数据结构
现在的数据表不单单只是存储数据,还有的是设计功能和快速处理数据的结构功能: 首先,我们在设计数据库的时候,我们要先分清楚,那些是要单纯的存储数据的(固定),然后再设计出来数据的表(流动) 你懂我意思吧 ...
- bootstrop设置背景图片自适应屏幕
如果不用bootstrop框架,想让背景图片自适应窗口大小,可以这样做: <style type="text/css"> html{height: %;} body.a ...
- [Inno Setup] 退出安装程序的两种方式
1. 完全静默的退出 procedure ExitProcess(exitCode:integer); external 'ExitProcess@kernel32.dll stdcall'; ... ...
- CentOS上安装配置 mongodb
CentOS 首先yum list mongo* 查看是否有关于mongo的安装包,检查后安装即可 mongo 分client端和server端,server启动db服务,client可以连接到s ...
- eclipse 创建maven项目失败
问题描述: eclipse 初次创建maven项目报错 可能是maven-archetype-quickstart:1.1.jar 包失效了或者没有? 有人说把这个jar包放在maven本地仓库里 我 ...
- slow-log 和bin-log相关参数介绍
1. slow-log show global status Slow_queries --------慢查询的次数,即查询的时间超过long_query_time设置的时间(不能修改) 配置文件 ...