【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 ...
随机推荐
- 2007 NOIP T1奖学金
奖学金(07NOIPT1): [题目描述] 某小学最近得到了一笔赞助,打算拿出其中一部分为学习成绩优秀的前5名学生发奖学金.期末,每个学生都有3门课的成绩:语文.数学.英语.先按总分从高到低排序,如果 ...
- bluecms v1.6 sp1 代码审计学习
前言 正式开始代码审计的学习,拓宽自己的知识面.代码审计学习的动力也是来自团队里的王叹之师傅,向王叹之师傅学习. 这里参考了一些前辈,师傅的复现经验和bluecms审计的心得 安装 install.p ...
- OAuth - 四种方式
OAuth 2.0 的标准是 RFC 6749 文件.该文件先解释了 OAuth 是什么. OAuth 引入了一个授权层,用来分离两种不同的角色:客户端和资源所有者.......资源所有者同意以后,资 ...
- Java 动态编译--DynamicCompiler
java 动态编译自己写过程的机会比较少,记录一下: package com.xzlf.dynamicCompile; import java.io.IOException; import java. ...
- linux内核第一宏 container_of
内核第一宏 list_entry()有着内核第一宏的美称,它被设计用来通过结构体成员的指针来返回结构体的指针.现在就让我们通过一步步的分析,来揭开它的神秘面纱,感受内核第一宏设计的精妙之处. 整理分析 ...
- CSS中“~”(波浪号)、“,”(逗号)、“+”(加号)、“>”(大于号)、“ ”(空格)详解
“~”:$('pre ~ brother')表示获取pre节点的后面的所有兄弟节点,相当于nextAll()方法: “+”:$('pre + nextbrother')表示获得pre节点的下一个兄弟节 ...
- opencv-7-鼠标绘制自定义图形
opencv-7-鼠标绘制自定义图形 opencvc++qt 开始之前 昨天写了具体的基本的图形绘制, 然后我们使用相应的函数接口进行调用, 便能够在图像上绘制出来相应的图形, 我们以图像绘制为例, ...
- java基于OpenCV的人脸识别
基于Java简单的人脸和人眼识别程序 使用这个程序之前必须先安装配置OpenCV详细教程见:https://www.cnblogs.com/prodigal-son/p/12768948.html 注 ...
- 翻译 - Kafka Streams 介绍(一)
2019独角兽企业重金招聘Python工程师标准>>> 资料 [原文地址](http://kafka.apache.org/11/documentation/streams/) 正文 ...
- MacBook Pro装Win7后喇叭没有声音
将MacBook的系统由XP改为Win7 64位后,发现喇叭没有声音了,装了bootcamp并升级到3.2版本都无济于事,google了下,发现还是驱动的问题,Win7下在设备管理器中看到声卡为Hig ...