传送门

题意简述:

给一个括号序列,要求支持:

  • 区间覆盖
  • 区间取负
  • 区间翻转
  • 查询把一个区间改成合法括号序列最少改几位

思路:

先考虑静态的时候如何维护答案。

显然把所有合法的都删掉之后序列长这样:

))...)))(((...(())...)))(((...(())...)))(((...((

于是可以给(((赋值成−1-1−1,)))赋值成111,这样只用维护前缀最大值aaa和后缀最小值bbb。

然后就可以知道答案是⌊a+12⌋+⌊−b+12⌋\left\lfloor\frac{a+1}2\right\rfloor+\left\lfloor\frac{-b+1}2\right\rfloor⌊2a+1​⌋+⌊2−b+1​⌋,然后由于a+b=a+b=a+b=这段区间的和,因此实际上只用维护aaa。

于是用一个fhqtreapfhq_treapfhqt​reap来支持上述修改即可。

代码:

#include<bits/stdc++.h>
#define fi first
#define se second
#define ri register int
#define gc getchar
using namespace std;
inline int read(){
    int ans=0;
    char ch=gc();
    while(!isdigit(ch))ch=gc();
    while(isdigit(ch))ans=((ans<<2)+ans<<1)+(ch^48),ch=gc();
    return ans;
}
typedef pair<int,int> pii;
const int N=1e5+5;
int n,m;
char s[N];
inline unsigned int randd(){
    static unsigned int x=23333;
    return x^=x<<13,x^=x>>17,x^=x<<5;
}
namespace fhq{
    #define lc son[p][0]
    #define rc son[p][1]
    int siz[N],son[N][2],rd[N],cov[N],ls[N],rs[N],sum[N],val[N],rt=0,tot=0;
    bool rev[N],inv[N];
    inline void pushup(int p){
        ls[p]=max(ls[lc],sum[lc]+val[p]+ls[rc]);
        rs[p]=max(rs[rc],sum[rc]+val[p]+rs[lc]);
        siz[p]=siz[lc]+1+siz[rc];
        sum[p]=sum[lc]+val[p]+sum[rc];
    }
    inline void pushcov(int p,int v){ls[p]=rs[p]=max((int)(inv[p]=rev[p]=0),sum[p]=((cov[p]=val[p]=v)*siz[p]));}
    inline void pushrev(int p){rev[p]^=1,swap(ls[p],rs[p]),swap(lc,rc);}
    inline void pushinv(int p){inv[p]^=1,ls[p]-=sum[p],rs[p]-=sum[p],swap(ls[p],rs[p]),val[p]=-val[p],sum[p]=-sum[p];}
    inline void pushdown(int p){
        if(cov[p])pushcov(lc,cov[p]),pushcov(rc,cov[p]),cov[p]=0;
        if(rev[p])pushrev(lc),pushrev(rc),rev[p]=0;
        if(inv[p])pushinv(lc),pushinv(rc),inv[p]=0;
    }
    inline int merge(int a,int b){
        if(!a||!b)return a+b;
        pushdown(a),pushdown(b);
        if(rd[a]<rd[b])return son[a][1]=merge(son[a][1],b),pushup(a),a;
        return son[b][0]=merge(a,son[b][0]),pushup(b),b;
    }
    inline pii split(int p,int k){
        if(!p)return pii(0,0);
        pii tmp;
        pushdown(p);
        if(siz[lc]>=k)return tmp=split(lc,k),lc=tmp.se,pushup(p),pii(tmp.fi,p);
        return tmp=split(rc,k-siz[lc]-1),rc=tmp.fi,pushup(p),pii(p,tmp.se);
    }
    inline void build(int&p,int l,int r){
        if(l>r)return;
        int mid=l+r>>1;
        p=++tot;
        val[p]=sum[p]=s[mid]=='('?-1:1,rd[p]=randd();
        if(l==r){ls[p]=rs[p]=max(0,val[p]),siz[p]=1;return;}
        build(lc,l,mid-1),build(rc,mid+1,r),pushup(p);
    }
    inline void cover(int l,int r,int v){
        pii x=split(rt,l-1),y=split(x.se,r-l+1);
        pushcov(y.fi,v),rt=merge(x.fi,merge(y.fi,y.se));
    }
    inline void reverse(int l,int r){
        pii x=split(rt,l-1),y=split(x.se,r-l+1);
        pushrev(y.fi),rt=merge(x.fi,merge(y.fi,y.se));
    }
    inline void invert(int l,int r){
        pii x=split(rt,l-1),y=split(x.se,r-l+1);
        pushinv(y.fi),rt=merge(x.fi,merge(y.fi,y.se));
    }
    inline void query(int l,int r){
        pii x=split(rt,l-1),y=split(x.se,r-l+1);
        cout<<((ls[y.fi]+1)>>1)+((ls[y.fi]-sum[y.fi]+1)>>1)<<'\n',rt=merge(x.fi,merge(y.fi,y.se));
    }
    #undef lc
    #undef rc
}
int main(){
    n=read(),m=read(),scanf("%s",s+1);
    fhq::build(fhq::rt,1,n);
    char s[10],t[2];
    for(ri l,r;m;--m){
        scanf("%s",s),l=read(),r=read();
        if(s[0]=='R'){
            scanf("%s",t);
            fhq::cover(l,r,t[0]=='('?-1:1);
        }
        if(s[0]=='Q')fhq::query(l,r);
        if(s[0]=='I')fhq::invert(l,r);
        if(s[0]=='S')fhq::reverse(l,r);
    }
    return 0;
}

2019.03.25 bzoj2329: [HNOI2011]括号修复(fhq_treap)的更多相关文章

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

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

  2. BZOJ2329 [HNOI2011]括号修复

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

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

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

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

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

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

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

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

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

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

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

  8. [2019.03.25]Linux中的查找

    TMUX天下第一 全世界所有用CLI Linux的人都应该用TMUX,我爱它! ======================== 以下是正文 ======================== Linu ...

  9. 【BZOJ2329】括号修复(Splay)

    [BZOJ2329]括号修复(Splay) 题面 BZOJ 洛谷 题解 本来想着用线段树来写 但是有一个区间翻转 所以不能用线段树了,就只能用平衡树 然后直接\(Splay\)就好了 注意一下几个标记 ...

随机推荐

  1. java学习--自定义类的实例的大小比较和排序

    我们知道Object类有一个equals方法,用于比较两个对象是否相等 我们只要在自定义类中重写了equals方法(若不重写就是比较两个实例的地址,相当于==)就可以用来比较该类的两个实例是否相等 问 ...

  2. C++并发编程学习笔记

    // //  main.cpp //  test1 // //  Created by sofard on 2018/12/27. //  Copyright © 2018年 dapshen. All ...

  3. task 异步 进程与线程的区别

    用Wait方法(会以同步的方式来执行),不用Wait则会以异步的方式来执行 要在主线程中等待后台线程执行完毕,可以使用Wait方法(会以同步的方式来执行).不用Wait则会以异步的方式来执行. Tas ...

  4. R语言-时间序列图

    1.时间序列图 plot()函数 > air<-read.csv("openair.csv") > plot(air$nox~as.Date(air$date,& ...

  5. 基于WebGL架构的3D可视化平台—实现小车行走路线演示

    小车行走路线演示New VS Old 刚接触ThingJS的时候,写的一个小车开进小区的演示,今天又看了教程中有movePath这个方法就重新写了一遍,其中也遇到了一些问题,尤其突出的问题就是小车过弯 ...

  6. 求1!+2!+3!+......+n!的和 -----C++-----

    #include<iostream> using namespace std; int function(int x) { ; ;i<=x;i++) sum=sum*i; retur ...

  7. Linux开始结束ping命令

    ctrl+c可以终止ping ctrl+z可以暂停ping,该暂停只是把进程放到后台去了,使用命令fg可以调出到前台来 通过以下命令可以设置次数: ping -c    10 (次数)  ip(域名) ...

  8. Game Engine Architecture 3

    [Game Engine Architecture 3] 1.Computing performance—typically measured in millions of instructions  ...

  9. 100-days: twenty-five

    Title: Want to be happy? Rent a Finnish person(芬兰人) to teach you how rent n.租金; 地租; (意见等的) 分裂,分歧; (衣 ...

  10. SpringCloud微服务基础

    1.传统项目架构  传统项目分为三层架构,将业务逻辑层.数据库访问层.控制层放入在一个项目中.适合于个人或者小团队开发,不适合大团队开发. 2.分布式项目架构(解耦方案) 根据业务需求进行拆分成N个子 ...