2329: [HNOI2011]括号修复

Time Limit: 40 Sec  Memory Limit: 128 MB
Submit: 1007  Solved: 476
[Submit][Status][Discuss]

Description

Input

Output

Sample Input

Sample Output

HINT

Source

Solution

一眼Splay么....关键是怎么维护...

一开始看错题了...想了很久歪路,其实想一下还是可以想到的么..

对于一个括号序列,首先合法的括号对是对询问答案没有贡献的,所以可以忽略,剩余的括号序列必然是形如$)*(*$的序列。

所以答案显然就是$\lfloor \frac{Num_{)}+1}{2} \rfloor + \lfloor \frac{Num{(}+1}{2} \rfloor$。

然后只需要Splay中维护左右起最大连续$($左右起最大连续$)$,即可得到答案。

考虑怎么维护这样的东西..括号序列一种最典型的表示方式可以认为是$+1-1$,这里也一样啊,不妨令$(=-1$,$)=+1$

这样就可以以一种类似最大最小连续子段和的方式去维护上述量了。

然后就是修改操作。

覆盖操作显然会清空翻转操作和取反操作,问题在于取反操作和覆盖操作的下放顺序会对结果有影响!!!(画图考虑一下)

所以值得注意的就是,在下放标记的时候,取反标记同样会使覆盖标记发生取反。(再画图考虑一下)

Code

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
inline int read()
{
int x=0,f=1; char ch=getchar();
while (ch<'0' || ch>'9') {if (ch=='-') f=-1; ch=getchar();}
while (ch>='0' && ch<='9') {x=x*10+ch-'0'; ch=getchar();}
return x*f;
}
#define MAXN 100010
#define INF 0x3fffffff
int N,M;
char s[MAXN]; namespace SplayTree{
#define lson(x) son[x][0]
#define rson(x) son[x][1]
int son[MAXN][2],fa[MAXN],root,sz;
int lmin[MAXN],rmin[MAXN],lmax[MAXN],rmax[MAXN],size[MAXN],sum[MAXN],val[MAXN];
int rev[MAXN],inv[MAXN],cov[MAXN];
inline void Newnode(int &x,int last,int v)
{
x=++sz;
size[x]=1;
val[x]=sum[x]=v;
if (v==1)
lmin[x]=rmin[x]=0,lmax[x]=rmax[x]=v;
else
lmin[x]=rmin[x]=v,lmax[x]=rmax[x]=0;
fa[x]=last; son[x][0]=son[x][1]=0;
}
inline int Right(int x) {return son[fa[x]][1]==x;}
inline void Update(int x)
{
if (!x) return;
sum[x]=sum[lson(x)]+sum[rson(x)]+val[x];
size[x]=size[lson(x)]+size[rson(x)]+1;
lmax[x]=max(lmax[lson(x)],sum[lson(x)]+val[x]+lmax[rson(x)]);
rmax[x]=max(rmax[rson(x)],sum[rson(x)]+val[x]+rmax[lson(x)]);
lmin[x]=min(lmin[lson(x)],sum[lson(x)]+val[x]+lmin[rson(x)]);
rmin[x]=min(rmin[rson(x)],sum[rson(x)]+val[x]+rmin[lson(x)]);
}
inline void Rev(int x)
{
if (!x) return;
rev[x]^=1;
swap(son[x][0],son[x][1]);
swap(lmax[x],rmax[x]),swap(lmin[x],rmin[x]);
}
inline void Cov(int x,int d)
{
if (!x) return;
cov[x]=d; rev[x]=0; inv[x]=0;
sum[x]=d*size[x]; val[x]=d;
if (d==1)
lmax[x]=rmax[x]=size[x],lmin[x]=rmin[x]=0;
else
lmax[x]=rmax[x]=0,lmin[x]=rmin[x]=-size[x];
}
inline void Inv(int x)
{
if (!x) return;
inv[x]^=1;
sum[x]=-sum[x]; val[x]=-val[x];
swap(lmax[x],lmin[x]),swap(rmax[x],rmin[x]);
lmax[x]=-lmax[x],rmax[x]=-rmax[x];
lmin[x]=-lmin[x],rmin[x]=-rmin[x];
cov[x]=-cov[x];
}
inline void Pushdown(int x)
{
if (rev[x]!=0)
Rev(son[x][0]),Rev(son[x][1]),rev[x]^=1;
if (inv[x]!=0)
Inv(son[x][0]),Inv(son[x][1]),inv[x]^=1;
if (cov[x]!=0)
Cov(son[x][0],cov[x]),Cov(son[x][1],cov[x]),cov[x]=0;
}
inline int Build(int l,int r,int last)
{
int mid=(l+r)>>1,x;
Newnode(x,last,s[mid]=='('? -1:1);
if (mid-1>=l) son[x][0]=Build(l,mid-1,x);
if (mid+1<=r) son[x][1]=Build(mid+1,r,x);
Update(x);
return x;
}
inline void Rotate(int x)
{
int y=fa[x],z=fa[y],w=Right(x);
Pushdown(y); Pushdown(x);
son[y][w]=son[x][w^1]; fa[son[y][w]]=y;
fa[y]=x; son[x][w^1]=y; fa[x]=z;
if (z) son[z][son[z][1]==y]=x;
Update(y); Update(x);
}
inline void Splay(int x,int tar)
{
for (int y; (y=fa[x])!=tar; Rotate(x))
if (fa[y]!=tar) Rotate(Right(x)==Right(y) ? y:x);
if (!tar) root=x;
}
inline int Find(int x,int k)
{
Pushdown(x);
if (size[son[x][0]]>=k) return Find(son[x][0],k);
if (size[son[x][0]]+1==k) return x;
return Find(son[x][1],k-size[son[x][0]]-1);
}
inline int Split(int l,int r)
{
int x=Find(root,l),y=Find(root,r+2);
Splay(x,0); Splay(y,root); return lson(rson(root));
}
inline void Cover(int l,int r,int d) {int x=Split(l,r); Cov(x,d);}
inline void Rever(int l,int r) {int x=Split(l,r); Rev(x);}
inline void Inver(int l,int r) {int x=Split(l,r); Inv(x);}
inline int Query(int l,int r) {int x=Split(l,r); return (lmax[x]+1)/2+(-rmin[x]+1)/2;}
}using namespace SplayTree; int main()
{
Newnode(root,0,0);
Newnode(son[root][1],root,0); N=read(),M=read();
scanf("%s",s+1); son[son[root][1]][0]=SplayTree::Build(1,N,son[root][1]); while (M--) {
char opt[10]; scanf("%s",opt+1);
int l=read(),r=read();
switch (opt[1]) {
case 'R' : scanf("%s",s+1); SplayTree::Cover(l,r,s[1]=='('? -1:1); break;
case 'Q' : printf("%d\n",SplayTree::Query(l,r)); break;
case 'S' : SplayTree::Rever(l,r); break;
case 'I' : SplayTree::Inver(l,r); break;
}
} return 0;
}

大过年的,写啥Splay啊....这道题又捯饬了两个多小时...真是自虐。

【BZOJ-2329&2209】括号修复&括号序列 Splay的更多相关文章

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

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

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

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

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

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

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

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

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

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

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

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

  7. BZOJ2329:[HNOI2011]括号修复

    浅谈\(splay\):https://www.cnblogs.com/AKMer/p/9979592.html 浅谈\(fhq\)_\(treap\):https://www.cnblogs.com ...

  8. 【BZOJ】2209: [Jsoi2011]括号序列(splay)

    http://www.lydsy.com/JudgeOnline/problem.php?id=2209 splay又犯逗........upd1那里的sum忘记赋值反............. 本题 ...

  9. bzoj 2209: [Jsoi2011]括号序列 splay

    2209: [Jsoi2011]括号序列 Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 833  Solved: 392[Submit][Status ...

随机推荐

  1. Redis记录-redis和memcached的区别

    1.Redis和Memcache都是将数据存放在内存中,都是内存数据库.不过memcache还可用于缓存其他东西,例如图片.视频等等: 2.Redis不仅仅支持简单的k/v类型的数据,同时还提供lis ...

  2. Anaconda+django写出第一个web app(十一)

    今天我们来学习给页面添加一个Sidebar,根据Sidebar跳转到相应的tutorial. 打开views.py,编辑single_slug函数: def single_slug(request, ...

  3. Anaconda+django写出第一个web app(八)

    今天来实现网站的登入和登出功能. 首先我们需要在urls.py中添加路径,注意此处的路径和在导航栏中设置的文字路径保持一致: from django.urls import path from . i ...

  4. FPGA学习笔记. DDS

    DDS原理 直接数字式频率合成器(Direct Digital Synthesizer) 频率计算公式 Fout = FW * Fclk / 2^N Fout 输出频率, Fw 频率控制字, N 位数 ...

  5. 01 uni-app框架学习:项目创建及底部导航栏tabBar配置

    1.创建一个项目类型选择uniapp 2. pages里新建3个页面如下 3.在pages.json中配置底部导航tabBar 效果展示:

  6. source insigh安装使用

    下载和安装: 最好去官网下载(http://www.sourceinsight.com/),最新版本是3.5. 第一次去六维下载了sourceinsight,免安装,但是打开后发现界面没有任何窗口,全 ...

  7. pymongo创建索引

    from database import db db_list = ["table1", "table2", "table3", " ...

  8. Spring bean 配置

    1.传统的创建对象的方式:JedisMall tardition=new JedisMall(); 这样是在程序运行时创建,表示当前模块已经不知不觉和new出的对象耦合了,而我们通常都是更高层次的抽象 ...

  9. 批量初始化数组和memset函数

    对于数组的初始化有一下三种方式: int  a[]={1,2,3,4,5} //通过判断初始化值得个数来却仍数组长度 int b[5]={1,2,3} //数组长度为5,可是初始值却只有三个,因此,不 ...

  10. docker 的简单操作

    一直说更博,但是一直在delay.... 最近一直用到docker,所以就总结一下吧! docker的介绍 Docker 是一个开源的应用容器引擎,基于 Go 语言 并遵从Apache2.0协议开源. ...