设\(nd[4]\)

0——多出来的右括号
1——多出来的左括号
2——取反后多出来的右括号
3——取反后多出来的左括号

这样一来

Swap: swap(0,3),swap(1,2),swap(sn[0],sn[1])
Invert: swap(0,2),swap(1,3),val[k]^=1
Replace: v<-siz[k],v+2^1<-siz[k],v^1<-0,v+2<-0

注意一下运算优先级就好

#include"cstdio"
#include"cstring"
#include"iostream"
#include"algorithm"
using namespace std; const int MAXN=1e5+5; int n,m,cnt,root;
char ch[MAXN];
int val[MAXN],rev[MAXN],sn[2][MAXN],siz[MAXN];
int nd[4][MAXN],tag3[MAXN];
bool tag1[MAXN],tag2[MAXN]; int cret(int v)
{
int tmp=++cnt;
siz[tmp]=1;
val[tmp]=v;
tag3[tmp]=-1;
nd[v][tmp]=nd[v+2^1][tmp]=1;
rev[tmp]=rand();
return tmp;
} void pushdown(int k)
{
if(tag3[k]!=-1){
if(sn[0][k]) nd[tag3[k]][sn[0][k]]=nd[tag3[k]+2^1][sn[0][k]]=siz[sn[0][k]],nd[tag3[k]^1][sn[0][k]]=nd[tag3[k]+2][sn[0][k]]=0,val[sn[0][k]]=tag3[sn[0][k]]=tag3[k],tag1[sn[0][k]]=tag2[sn[0][k]]=0;
if(sn[1][k]) nd[tag3[k]][sn[1][k]]=nd[tag3[k]+2^1][sn[1][k]]=siz[sn[1][k]],nd[tag3[k]^1][sn[1][k]]=nd[tag3[k]+2][sn[1][k]]=0,val[sn[1][k]]=tag3[sn[1][k]]=tag3[k],tag1[sn[1][k]]=tag2[sn[1][k]]=0;
}if(tag2[k]){
swap(sn[0][k],sn[1][k]);
if(sn[0][k]) swap(nd[0][sn[0][k]],nd[3][sn[0][k]]),swap(nd[1][sn[0][k]],nd[2][sn[0][k]]),tag2[sn[0][k]]^=1;
if(sn[1][k]) swap(nd[0][sn[1][k]],nd[3][sn[1][k]]),swap(nd[1][sn[1][k]],nd[2][sn[1][k]]),tag2[sn[1][k]]^=1;
}if(tag1[k]){
if(sn[0][k]) swap(nd[0][sn[0][k]],nd[2][sn[0][k]]),swap(nd[1][sn[0][k]],nd[3][sn[0][k]]),tag1[sn[0][k]]^=1,val[sn[0][k]]^=1;
if(sn[1][k]) swap(nd[0][sn[1][k]],nd[2][sn[1][k]]),swap(nd[1][sn[1][k]],nd[3][sn[1][k]]),tag1[sn[1][k]]^=1,val[sn[1][k]]^=1;
}tag1[k]=tag2[k]=0;
tag3[k]=-1;
return;
} void pushup(int k)
{
siz[k]=siz[sn[0][k]]+siz[sn[1][k]]+1;
nd[0][k]=nd[0][sn[0][k]];
nd[1][k]=nd[1][sn[1][k]];
int tmp=nd[1][sn[0][k]]-nd[0][sn[1][k]];
tmp+=val[k]?1:-1;
if(tmp>0) nd[1][k]+=tmp;
else nd[0][k]-=tmp; nd[2][k]=nd[2][sn[0][k]];
nd[3][k]=nd[3][sn[1][k]];
tmp=nd[3][sn[0][k]]-nd[2][sn[1][k]];
tmp+=val[k]?-1:1;
if(tmp>0) nd[3][k]+=tmp;
else nd[2][k]-=tmp;
return;
} void dro(int k,int v,int &x,int &y)
{
if(!k){x=y=0;return;}
pushdown(k);
if(siz[sn[0][k]]<v) x=k,dro(sn[1][k],v-siz[sn[0][k]]-1,sn[1][k],y);
else y=k,dro(sn[0][k],v,x,sn[0][k]);
pushup(k);
return;
} int un(int x,int y)
{
if(!x||!y) return x|y;
if(rev[x]<rev[y]){
pushdown(x);
sn[1][x]=un(sn[1][x],y);
pushup(x);
return x;
}pushdown(y);
sn[0][y]=un(x,sn[0][y]);
pushup(y);
return y;
} void slv0(int l,int r)
{
int x,y,z;
dro(root,r,x,z);
dro(x,l-1,x,y);
printf("%d\n",(nd[0][y]+1)/2+(nd[1][y]+1)/2);
root=un(un(x,y),z);
return;
} void slv1(int l,int r)
{
int x,y,z;
dro(root,r,x,z);
dro(x,l-1,x,y);
tag1[y]^=1;val[y]^=1;
pushdown(y);pushup(y);
root=un(un(x,y),z);
return;
} void slv2(int l,int r)
{
int x,y,z;
dro(root,r,x,z);
dro(x,l-1,x,y);
tag2[y]^=1;
pushdown(y);pushup(y);
root=un(un(x,y),z);
return;
} void slv3(int l,int r,int kd)
{
int x,y,z;
dro(root,r,x,z);
dro(x,l-1,x,y);
tag3[y]=val[y]=kd;tag1[y]=tag2[y]=0;
pushdown(y);pushup(y);
root=un(un(x,y),z);
return;
} int main()
{
scanf("%d%d",&n,&m);
scanf("%s",ch+1);
for(int i=1;i<=n;++i) root=un(root,cret(ch[i]=='('));
while(m--){
int l,r;
scanf("%s%d%d",ch+1,&l,&r);
if(ch[1]=='Q') slv0(l,r);
else if(ch[1]=='I') slv1(l,r);
else if(ch[1]=='S') slv2(l,r);
else scanf("%s",ch+1),slv3(l,r,ch[1]=='(');
}return 0;
}

[HNOI2011]括号修复的更多相关文章

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

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

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

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

  3. bzoj千题计划222:bzoj2329: [HNOI2011]括号修复(fhq treap)

    http://www.lydsy.com/JudgeOnline/problem.php?id=2329 需要改变的括号序列一定长这样 :)))((( 最少改变次数= 多余的‘)’/2 [上取整] + ...

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

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

  5. BZOJ2329 [HNOI2011]括号修复

    把左括号看做$1$,右括号看做$-1$,于是查询操作等于查询一个区间左边右边最大(最小)子段和 支持区间翻转,反转,覆盖操作...注意如果有覆盖操作,之前的操作全部作废了...于是在下传标记的时候要最 ...

  6. BZOJ 2329: [HNOI2011]括号修复 [splay 括号]

    题目描述 一个合法的括号序列是这样定义的: 空串是合法的. 如果字符串 S 是合法的,则(S)也是合法的. 如果字符串 A 和 B 是合法的,则 AB 也是合法的. 现在给你一个长度为 N 的由‘(' ...

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

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

  8. 2019.03.25 bzoj2329: [HNOI2011]括号修复(fhq_treap)

    传送门 题意简述: 给一个括号序列,要求支持: 区间覆盖 区间取负 区间翻转 查询把一个区间改成合法括号序列最少改几位 思路: 先考虑静态的时候如何维护答案. 显然把所有合法的都删掉之后序列长这样: ...

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

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

随机推荐

  1. Linq to xml修改CDATA节点值

    增加节点时,我们是这样写的: xop.Document.Element("messages").Add( new XElement("message", new ...

  2. 组件基础(参数校验和动态组件、v-once)—Vue学习笔记

    最最最后一点关于组件传值的问题. 提醒:本篇内容请使用Vue.js开发版!(附带完成的警告和提示) 1.组件的参数校验 父组件向子组件传值,子组件可以决定传值的一些限制. 比如,子组件指向接收Stri ...

  3. MySQL 排名统计(常用功能函数)

    select actor_id,@curr_cnt:=cnt as cnt , ,@rank) as rank, @prev_cnt:=@curr_cnt as dummy from( select ...

  4. SQL将原始数据进行MD5加密转存

    先来描述下情况吧,首先有一批用户之前批量录入后默认的密码为6个8然后进行MD5加密后进行存储的,现在需要对其更改根据用户身份证号后6位作为密码. 1.首先发现我们sqlserver05以上的版本是自带 ...

  5. Log4j 相关

    Log4j(Log for Java) Log4j是Apache提供的一种专门用于Java程序记录日志的工具,是目前主流的开发日志技术. 日志的作用: 1.记录系统运行过程中的重要运行信息 a) 付费 ...

  6. JavaScript -- Table

    -----048-Table.html----- <!DOCTYPE html> <html> <head> <meta http-equiv="c ...

  7. mpvue使用scroll-view实现图片横向滑动

    html代码 <div class="swiper-home"> <scroll-view scroll-x="true" style=&qu ...

  8. bigdata-02-hadoop2.8.4-resourceHA安装

    1, 电脑环境准备 1), 关闭selinux vim /etc/selinux/config SELINUX=disabled 2), 时间同步 yum -y install chrony 修改时间 ...

  9. BackgroundWorker简单实用(简便的异步操作)

    微软提供了一个快捷使用多线程的帮助类BackgroundWorker,能够快速创建一个新的线程,并能报告进度,暂停,以及在线程完成后处理别的任务. 1.BackgroundWorker类介绍 1.1. ...

  10. 前端自动构建工具Gulp入门

    基于nodeJs:通过不同插件能自动完成一系列动作,比如压缩js/css/img.解析模版标签.解析less等: 一.安装gulp 安装nodeJs 打开Node.js command prompt ...