题目描述

一个合法的括号序列是这样定义的:

  1. 空串是合法的。

  2. 如果字符串 S 是合法的,则(S)也是合法的。

  3. 如果字符串 A 和 B 是合法的,则 AB 也是合法的。

现在给你一个长度为 N 的由‘('和‘)'组成的字符串,位置标号从 1 到 N。对这个字符串有下列四种操作:

  1. Replace a b c:将[a,b]之间的所有括号改成 c。例如:假设原来的字符串为:))())())(,那么执行操作 Replace 2 7 ( 后原来的字符串变为:)(((((()(。

  2. Swap a b:将[a,b]之间的字符串翻转。例如:假设原来的字符串为:))())())(,那么执行操作 Swap 3 5 后原来的字符串变为:))))(())(。

  3. Invert a b:将[a,b]之间的‘(’变成‘)’,‘)’变成‘(’。例如:假设原来的字符串为:))())())(,那么执行操作 Invert 4 8 后原来的字符串变为:))((()(((。

  4. Query a b:询问[a,b]之间的字符串至少要改变多少位才能变成合法的括号序列。改变某位是指将该位的‘(’变成‘)’或‘)’变成‘(’。注意执行操作 Query 并不改变当前的括号序列。例如:假设原来的字符串为:))())())(,那么执行操作 Query 3 6

的结果为 2,因为要将位置 5 的‘)’变成‘(’并将位置 6 的‘(’变成‘)’。

输入输出格式

输入格式:

从文件input.txt中读入数据,输入文件的第一行是用空格隔开的两个正整数N和M,分别表示字符串的长度和将执行的操作个数。第二行是长度为N的初始字符串S。接下来的M行是将依次执行的M个操作,其中操作名与操作数之间以及相邻操作数之间均用空格隔开。30%的数据满足

N,M≤3000。100%的数据满足N,M≤100000。

输出格式:

输出文件 output.txt 包含 T 行,其中 T 是输入的将执行的 M 个操作中 Query 操作出现的次数。Query 操作的每次出现依次对应输出文件中的一行,该行只有一个非负整数,表示执行对应 Query 操作的结果,即:所指字符串至少要改变多少位才能变成合法的括号序列。输入数据

保证问题有解。

输入输出样例

输入样例#1:

4 5
((((
Replace 1 2 )
Query 1 2
Swap 2 3
Invert 3 4
Query 1 4
输出样例#1:

1
2

说明

样例解释:输入中有2个Query操作,所以输出有2行。执行第一个Query操作时的括号序列为))((,因改变第1位可使[1,2]之间的字符串变成合法的括号序列,故输出的第一行为1。执行第二个Query操作时的括号序列为)((),因要改变第1位和第2位才能使[1,4]之间的字符串变成合法的括号序列,故输出的第二行为2。


去luogu偷题面


多了区间修改操作..................

区间修改后要把反转翻转标记清空

反转标记需要反转修改标记

PS:luogu时限不科学,要强开O2才过

#pragma GCC optimize(2)
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
typedef long long ll;
#define lc t[x].ch[0]
#define rc t[x].ch[1]
#define pa t[x].fa
const int N=1e5+,INF=1e9;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-; c=getchar();}
while(c>=''&&c<=''){x=x*+c-''; c=getchar();}
return x*f;
}
int n,Q,l,r,a[N];
char s[N],op[];
struct node{
int ch[],fa,size,v,sum,lmx,lmn,rmx,rmn,rev,flp,tag;
node():fa(){ch[]=ch[]=;}
}t[N];
int root;
inline int wh(int x){return t[pa].ch[]==x;}
inline void update(int x){
if(!x) return;
t[x].size=t[lc].size+t[rc].size+;
t[x].sum=t[lc].sum+t[rc].sum+t[x].v;
t[x].lmx=max(t[lc].lmx,max(t[lc].sum+t[x].v,t[lc].sum+t[x].v+t[rc].lmx));
t[x].lmn=min(t[lc].lmn,min(t[lc].sum+t[x].v,t[lc].sum+t[x].v+t[rc].lmn));
t[x].rmx=max(t[rc].rmx,max(t[rc].sum+t[x].v,t[rc].sum+t[x].v+t[lc].rmx));
t[x].rmn=min(t[rc].rmn,min(t[rc].sum+t[x].v,t[rc].sum+t[x].v+t[lc].rmn));
}
inline void rever(int x){
t[x].rev^=;
swap(lc,rc);
swap(t[x].lmx,t[x].rmx);
swap(t[x].lmn,t[x].rmn);
}
inline void flip(int x){
t[x].flp^=;
t[x].sum=-t[x].sum;t[x].v=-t[x].v;
t[x].lmx=-t[x].lmx;t[x].lmn=-t[x].lmn;
swap(t[x].lmx,t[x].lmn);
t[x].rmx=-t[x].rmx;t[x].rmn=-t[x].rmn;
swap(t[x].rmx,t[x].rmn);
t[x].tag=-t[x].tag;
}
inline void paint(int x,int d){
t[x].rev=t[x].flp=;
t[x].v=t[x].tag=d;t[x].sum=t[x].size*d;
t[x].lmx=t[x].rmx=max(d,t[x].sum);
t[x].lmn=t[x].rmn=min(d,t[x].sum);
}
inline void pushDown(int x){
if(t[x].rev){
if(lc) rever(lc);
if(rc) rever(rc);
t[x].rev=;
}
if(t[x].flp){
if(lc) flip(lc);
if(rc) flip(rc);
t[x].flp=;
}
if(t[x].tag){
if(lc) paint(lc,t[x].tag);
if(rc) paint(rc,t[x].tag);
t[x].tag=;
}
} inline void rotate(int x){
int f=t[x].fa,g=t[f].fa,c=wh(x);
if(g) t[g].ch[wh(f)]=x;t[x].fa=g;
t[f].ch[c]=t[x].ch[c^];t[t[f].ch[c]].fa=f;
t[x].ch[c^]=f;t[f].fa=x;
update(f);update(x);
}
inline void splay(int x,int tar){
for(;pa!=tar;rotate(x))
if(t[pa].fa!=tar) rotate(wh(x)==wh(pa)?pa:x);
if(tar==) root=x;
} int build(int l,int r,int f){//printf("build %d %d %d\n",l,r,f);
if(l>r) return ;
int x=(l+r)>>;
lc=build(l,x-,x);rc=build(x+,r,x);
t[x].fa=f;
t[x].rev=t[x].flp=t[x].tag=;
t[x].v=a[x];//not need
update(x);//printf("get %d %d %d %d %d %d %d %d %d\n",x,l,r,t[x].v,t[x].sum,t[x].lmx,t[x].lmn,t[x].rmx,t[x].rmn);
return x;
}
inline int kth(int k){//printf("kth %d\n",k);
int x=root,ls=;
while(x){
pushDown(x);
int _=ls+t[lc].size;//printf("size %d %d\n",x,_);
if(_<k&&k<=_+) return x;
else if(k<=_) x=lc;
else ls=_+,x=rc;
}
return ;
}
void Query(int l,int r){//printf("query %d %d\n",l,r);
int f=kth(l);splay(f,);
int x=kth(r+);splay(x,f);
int a=t[lc].lmx,b=t[lc].rmn;//printf("hi %d %d %d %d %d\n",f,x,lc,a,b);
printf("%d\n",(a+)/+(-b+)/);
}
void Flip(int l,int r){
int f=kth(l);splay(f,);
int x=kth(r+);splay(x,f);
flip(lc);update(x);update(f);
}
void Rever(int l,int r){
int f=kth(l);splay(f,);
int x=kth(r+);splay(x,f);
rever(lc);update(x);update(f);
}
void Cover(int l,int r){
char c[];scanf("%s",c);
int d= c[]=='('?-:;
int f=kth(l);splay(f,);
int x=kth(r+);splay(x,f);
paint(lc,d);update(x);update(f);
}
void print(int x){
pushDown(x);
if(lc) print(lc);
printf("%c ",t[x].v==-?'(':')');
if(rc) print(rc);
}
int main(){
//freopen("in.txt","r",stdin);
n=read();Q=read();
scanf("%s",s+);
for(int i=;i<=n;i++) a[i+]=s[i]=='('?-:;
//for(int i=1;i<=n+2;i++) printf("%d ",a[i]);puts("");
t[].lmn=t[].rmn=INF;
t[].lmx=t[].rmx=-INF;
root=build(,n+,root);
while(Q--){//print(root);puts(" end");
scanf("%s",op);l=read();r=read();
if(op[]=='R') Cover(l,r);
if(op[]=='S') Rever(l,r);
if(op[]=='I') Flip(l,r);
if(op[]=='Q') Query(l,r);
}
}

BZOJ 2329: [HNOI2011]括号修复 [splay 括号]的更多相关文章

  1. ●BZOJ 2329 [HNOI2011]括号修复.cpp

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=2329 题解: Splay 类似 BZOJ 2329 [HNOI2011]括号修复 只是多了一 ...

  2. 【BZOJ2329/2209】[HNOI2011]括号修复/[Jsoi2011]括号序列 Splay

    [BZOJ2329/2209][HNOI2011]括号修复/[Jsoi2011]括号序列 题解:我们的Splay每个节点维护如下东西:左边有多少多余的右括号,右边有多少多余的左括号,同时为了反转操作, ...

  3. BZOJ 2329: [HNOI2011]括号修复( splay )

    把括号序列后一定是))))((((这种形式的..所以维护一个最大前缀和l, 最大后缀和r就可以了..答案就是(l+1)/2+(r+1)/2...用splay维护,O(NlogN). 其实还是挺好写的, ...

  4. [HNOI2011][bzoj 2329] 括号修复 [splay+前缀和]

    题面: http://www.lydsy.com/JudgeOnline/problem.php?id=2329 思路: 显然,操作4中输出补全的最小费用是关键 这决定了我们不可能在splay上只维护 ...

  5. BZOJ 2329/2209 [HNOI2011]括号修复 (splay)

    题目大意: 让你维护一个括号序列,支持 1.区间修改为同一种括号 2.区间内所有括号都反转 3.翻转整个区间,括号的方向不变 4.查询把某段区间变为合法的括号序列,至少需要修改多少次括号 给跪了,足足 ...

  6. 洛谷 P3215 [HNOI2011]括号修复 / [JSOI2011]括号序列(fhq-treap)

    题目链接 题意:有一个长度为 \(n\) 的括号序列,你需要支持以下操作: 将 \([l,r]\) 中所有括号变为 \(c\) 将 \([l,r]\) 区间翻转 将 \([l,r]\) 区间中左括号变 ...

  7. BZOJ2329 HNOI2011 括号修复 splay+贪心

    找平衡树练习题的时候发现了这道神题,可以说这道题是近几年单考splay的巅峰之作了. 题目大意:给出括号序列,实现区间翻转,区间反转和区间更改.查询区间最少要用几次才能改成合法序列. 分析: 首先我们 ...

  8. 【bzoj2329】[HNOI2011]括号修复 Splay

    题目描述 题解 Splay 由于有区间反转操作,因此考虑Splay. 考虑答案:缩完括号序列后剩下的一定是 $a$ 个')'+ $b$ 个'(',容易发现答案等于 $\lceil\frac a2\rc ...

  9. BZOJ2329: [HNOI2011]括号修复(Splay)

    解题思路: Replace.Swap.Invert都可以使用Splay完美解决(只需要解决一下标记冲突就好了). 最后只需要统计左右括号冲突就好了. 相当于动态统计最大前缀合和最小后缀和. 因为支持翻 ...

随机推荐

  1. gulp + es6 + babel+ angular 搭建环境并实现简单的路由

    1.ECMAscript 6的语法糖面临的唯一问题就是浏览器兼容的问题,使得很多程序员望而怯步. 2.babel的作用就是将es6的语法编译成es5被浏览器所识别.这样就可以任性的使用es6了. 3. ...

  2. spring是什么???

    1.是一个容器 2.用于降低代码间的耦合度3.根据不同的代码采用ioc和aop两种技术解耦合...

  3. mysql 批量更新与批量更新多条记录的不同值实现方法

    作者: 字体:[增加 减小] 类型:转载 时间:2013-10-02 我要评论 在mysql中批量更新我们可能使用update,replace into来操作,下面小编来给各位同学详细介绍mysql ...

  4. ngRx 官方示例分析 - 5. components

    组件通过标准的 Input 和 Output 进行操作,并不直接访问 store. /app/components/book-authors.ts import { Component, Input ...

  5. href

    <a href="#"></a>点击浏览器会跳转地址栏加了#:<a href=""></a>点击会打开当前网页所 ...

  6. UE4/Unity3D中同时捕获多高清摄像头的高效插件

    本文主要讲实现过程的一些坑. 先说下要实现的目标,主要功能在UE4/Unity中都要用,能同时捕获多个摄像头,并且捕获的图片要达到1080p25桢上,并且需要经过复杂的图片处理后丢给UE4/Unity ...

  7. es6重点笔记:Symbol,Set,Map,Proxy,Reflect

    一,Symbol 原始数据类型,不是对象,它是JavaScript第七种数据类型,表示独一无二的值.Symbol是通过Symbol函数生成的: let s = Symbol(); typeof s / ...

  8. 006-接收键盘的输入(read)

    read  -ptns   变量名 -p 在等待read输入的时候,显示的提示信息 -t 秒数,read等待用户输入的时间 -n read接收用户输入的字符数,只接收指定字符数,就会执行 -s 隐藏输 ...

  9. JavaScript ES6 module 模块

    在使用JavaScript开发大型项目时,模块开发概念是一个必须考虑的问题.其目的就是通过命名空间对各类业务对象进行一定的封装,防止命名冲突. 本篇着重介绍ES6 module中的export和imp ...

  10. linux_磁盘分区

    分区并没有数据内容只是改变分区表,保存在0磁头,0磁道1扇区除MBR引导后64bytes中,只能有4个组分区,4个以上要一个扩展分区 引导MBR,保存在446字节中 磁盘想要存放数据,首先要分区,可以 ...