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. vue element-ui 实现点击查看审核记录

    <el-dialog title="审核信息" :visible.sync="seeVisible" width="30%" :bef ...

  2. bzoj千题计划265:bzoj4873: [六省联考2017]寿司餐厅

    http://www.lydsy.com/JudgeOnline/problem.php?id=4873 选a必选b,a依赖于b 最大权闭合子图模型 构图: 1.源点 向 正美味度区间 连 流量为 美 ...

  3. c++刷题(33/100)笔试题1

    笔试总共2小时,三道题,时间挺充裕的,但是最后只做了一道,原因在于自己很浮躁,不审题,不仔细思考.没过的两道都是稍微改一下代码就能过,但是没过就是没过,要引以为戒 题目1: 小W有一个电子时钟用于显示 ...

  4. [转]LaTex常用数学符号整理

    转载自 http://blog.csdn.net/ying_xu/article/details/51240291 (自己保存方便查阅,侵删) 另一个网站 Markdown 添加 Latex 数学公式 ...

  5. Mini Twitter

    Implement a simple twitter. Support the following method: postTweet(user_id, tweet_text). Post a twe ...

  6. BAT修改文本内容

    @echo off (for /f "delims=" %%a in (文件名) do ( set "str=%%a" setlocal enabledelay ...

  7. Gitlab的安装汉化及问题解决

    Gitlab的安装汉化及问题解决(2017/12/14目前版本为10.2.4) 一.前言 Gitlab需要安装的包太TM多了,源码安装能愁死个人,一直出错,后来发现几行命令就装的真是遇到的新大陆一样. ...

  8. Ansible Tower系列 二(安装 Tower)【转】

    文档:http://docs.ansible.com/ansible-tower/ 安装前检查 python版本为2.6 保持网络畅通 内存预留充足 安装用户为root 软件下载 下载地址:http: ...

  9. C++获取文件夹下所有文件名

    查找文件需要一个结构体和几个函数.结构体为struct _finddata_t,函数为_findfirst.findnext和_findclose. struct _finddata_t 这个结构体是 ...

  10. 大数据系列之分布式计算批处理引擎MapReduce实践-排序

    清明刚过,该来学习点新的知识点了. 上次说到关于MapReduce对于文本中词频的统计使用WordCount.如果还有同学不熟悉的可以参考博文大数据系列之分布式计算批处理引擎MapReduce实践. ...