【BZOJ-2329&2209】括号修复&括号序列 Splay
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的更多相关文章
- BZOJ 2329/2209 [HNOI2011]括号修复 (splay)
题目大意: 让你维护一个括号序列,支持 1.区间修改为同一种括号 2.区间内所有括号都反转 3.翻转整个区间,括号的方向不变 4.查询把某段区间变为合法的括号序列,至少需要修改多少次括号 给跪了,足足 ...
- BZOJ 2329: [HNOI2011]括号修复( splay )
把括号序列后一定是))))((((这种形式的..所以维护一个最大前缀和l, 最大后缀和r就可以了..答案就是(l+1)/2+(r+1)/2...用splay维护,O(NlogN). 其实还是挺好写的, ...
- 【BZOJ2329/2209】[HNOI2011]括号修复/[Jsoi2011]括号序列 Splay
[BZOJ2329/2209][HNOI2011]括号修复/[Jsoi2011]括号序列 题解:我们的Splay每个节点维护如下东西:左边有多少多余的右括号,右边有多少多余的左括号,同时为了反转操作, ...
- ●BZOJ 2329 [HNOI2011]括号修复.cpp
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=2329 题解: Splay 类似 BZOJ 2329 [HNOI2011]括号修复 只是多了一 ...
- bzoj千题计划222:bzoj2329: [HNOI2011]括号修复(fhq treap)
http://www.lydsy.com/JudgeOnline/problem.php?id=2329 需要改变的括号序列一定长这样 :)))((( 最少改变次数= 多余的‘)’/2 [上取整] + ...
- 【BZOJ2329】括号修复(Splay)
[BZOJ2329]括号修复(Splay) 题面 BZOJ 洛谷 题解 本来想着用线段树来写 但是有一个区间翻转 所以不能用线段树了,就只能用平衡树 然后直接\(Splay\)就好了 注意一下几个标记 ...
- BZOJ2329:[HNOI2011]括号修复
浅谈\(splay\):https://www.cnblogs.com/AKMer/p/9979592.html 浅谈\(fhq\)_\(treap\):https://www.cnblogs.com ...
- 【BZOJ】2209: [Jsoi2011]括号序列(splay)
http://www.lydsy.com/JudgeOnline/problem.php?id=2209 splay又犯逗........upd1那里的sum忘记赋值反............. 本题 ...
- bzoj 2209: [Jsoi2011]括号序列 splay
2209: [Jsoi2011]括号序列 Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 833 Solved: 392[Submit][Status ...
随机推荐
- angular模块
深入浅析AngularJS中的模块 模块是AngularJS应用程序的一个组成部分,模块可以是一个Controller.Service服务.Filter过滤器.directive指令,这些都属于模块. ...
- Spark记录-本地Spark读取Hive数据简单例子
注意:将mysql的驱动包拷贝到spark/lib下,将hive-site.xml拷贝到项目resources下,远程调试不要使用主机名 import org.apache.spark._ impor ...
- 阿里云(一)云存储OSS的命令行osscmd的安装和使用
一.安装Python 在Linux Shell里验证Python版本: $ python -V Python 2.7.10 二.安装OSScmd SDK osscmd是基于python 2.5.4(其 ...
- 程序员与HR博弈之:有城府的表达你的兴趣爱好
“面试”这个过程说简单其实也能很简单.譬如急需招某种技能的单位会因为你拥有某方面的经验或特长立马录取你,哪怕你其他方面表现的很“烂”. 从广义上来讲,很多公司尤其是大中型公司的招聘,并不是因为急缺某岗 ...
- 天气窗件展示 -一个HTML5 地理位置应用的例子
定位及地理位置信息是LBS应用的核心,和定位功能有所不同的是地理位置信息更关注如何得到有意义的信息.(例如一条街道的地址) 从这边文章里你会学到HTML5地理位置信息的各种功能.它 ...
- (FFT) A * B Problem Plus
题目链接:https://cn.vjudge.net/contest/280041#problem/F 题目大意:给你两个数,求这俩数相乘的结果.(长度最长5000) 具体思路:硬算肯定是不行的,比如 ...
- [Alg::Trick]小白鼠找毒酒
题目来源:牛客网 https://www.nowcoder.com/questionTerminal/c26c4e43c77440ee9497b20118871bf1 8瓶酒一瓶有毒,用人测试.每次测 ...
- 数组中累加和小于等于k的最长子数组
问题描述: 给定一个无序数组arr,其中元素可正.可负.可0,给定一个整数 k.求arr所有的子数组中累加和小于或等于k的最长子数组长度.例如:arr=[3,-2,-4,0,6],k=-2,相加和小于 ...
- SqlServer查看表、存储过程、耗时查询、当前进程、开销较大的语句
--查看数据库中表的语句 SELECT s2.dbid , DB_NAME(s2.dbid) AS [数据库名] , --s1.sql_handle , ( SELECT TOP 1 SUBSTRIN ...
- 阿里云对象存储 OSS,不使用主账号,使用子账号来访问存储内容
https://help.aliyun.com/document_detail/31932.html?spm=5176.doc31929.2.5.R7sEzr 这个示例从一个没有任何Bucket的阿里 ...