【BZOJ2329】括号修复(Splay)
【BZOJ2329】括号修复(Splay)
题面
题解
本来想着用线段树来写
但是有一个区间翻转
所以不能用线段树了,就只能用平衡树
然后直接\(Splay\)就好了
注意一下几个标记的下放问题
这种数据结构真的没有什么思路可言。。
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 111111
#define ls (t[x].ch[0])
#define rs (t[x].ch[1])
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
struct Node
{
int ch[2],ff,size;
int lm,rm,lg,rg,sum,v;
int rev,eql,inv;
}t[MAX];
int rt;
void pushup(int x)
{
t[x].lm=max(t[ls].lm,t[ls].sum+t[x].v+t[rs].lm);
t[x].rm=max(t[rs].rm,t[rs].sum+t[x].v+t[ls].rm);
t[x].lg=min(t[ls].lg,t[ls].sum+t[x].v+t[rs].lg);
t[x].rg=min(t[rs].rg,t[rs].sum+t[x].v+t[ls].rg);
t[x].sum=t[ls].sum+t[rs].sum+t[x].v;
t[x].size=t[ls].size+t[rs].size+1;
}
void rotate(int x)
{
int y=t[x].ff,z=t[y].ff;
int k=t[y].ch[1]==x;
if(z)t[z].ch[t[z].ch[1]==y]=x;t[x].ff=z;
t[y].ch[k]=t[x].ch[k^1];t[t[x].ch[k^1]].ff=y;
t[x].ch[k^1]=y;t[y].ff=x;
pushup(y);
}
void putrev(int x)
{
if(!x)return;
swap(t[x].lg,t[x].rg);
swap(t[x].lm,t[x].rm);
swap(ls,rs);t[x].rev^=1;
}
void putinv(int x)
{
if(!x)return;
t[x].v*=-1;t[x].inv^=1;t[x].sum*=-1;
swap(t[x].lm,t[x].lg);swap(t[x].rm,t[x].rg);
t[x].lm*=-1;t[x].lg*=-1;t[x].rm*=-1;t[x].rg*=-1;
}
void puteql(int x,int w)
{
if(!x)return;
t[x].eql=t[x].v=w;t[x].sum=t[x].size*w;
t[x].rev=t[x].inv=0;
t[x].lm=t[x].rm=max(0,t[x].sum);
t[x].lg=t[x].rg=min(0,t[x].sum);
}
void pushdown(int x)
{
if(t[x].eql)puteql(ls,t[x].eql),puteql(rs,t[x].eql),t[x].eql=0;
if(t[x].rev)putrev(ls),putrev(rs),t[x].rev^=1;
if(t[x].inv)putinv(ls),putinv(rs),t[x].inv^=1;
}
void Splay(int x,int goal)
{
while(t[x].ff!=goal)
{
int y=t[x].ff,z=t[y].ff;
if(z!=goal)
(t[y].ch[0]==x)^(t[z].ch[0]==y)?rotate(x):rotate(y);
rotate(x);
}
if(!goal)rt=x;pushup(x);
}
int Kth(int x,int k)
{
while(233)
{
pushdown(x);
if(k<=t[ls].size)x=ls;
else if(k==t[ls].size+1)return x;
else k-=t[ls].size+1,x=rs;
}
}
int split(int l,int r)
{
int x=Kth(rt,l);Splay(x,0);
int y=Kth(rt,r+2);Splay(y,x);
return t[y].ch[0];
}
void Build(int l,int r,int ff)
{
if(l>r)return;
int mid=(l+r)>>1;
t[mid].ff=ff;
if(ff)t[ff].ch[ff<mid]=mid;
else rt=mid;
Build(l,mid-1,mid);Build(mid+1,r,mid);
pushup(mid);
}
char S[MAX];
int n,m;
int main()
{
n=read();m=read();
scanf("%s",S+1);
for(int i=1;i<=n;++i)t[i+1].v=(S[i]=='('?-1:1);
Build(1,n+2,0);
char ch[10];
while(m--)
{
scanf("%s",ch);
int l=read(),r=read(),x=split(l,r);
if(ch[0]=='Q')printf("%d\n",(t[x].lm+1)/2+(1-t[x].rg)/2);
else if(ch[0]=='S')putrev(x);
else if(ch[0]=='R')
{
scanf("%s",ch);int w=(ch[0]=='('?-1:1);
puteql(x,w);
}
else putinv(x);
}
return 0;
}
【BZOJ2329】括号修复(Splay)的更多相关文章
- BZOJ 2329: [HNOI2011]括号修复( splay )
把括号序列后一定是))))((((这种形式的..所以维护一个最大前缀和l, 最大后缀和r就可以了..答案就是(l+1)/2+(r+1)/2...用splay维护,O(NlogN). 其实还是挺好写的, ...
- BZOJ2329 HNOI2011 括号修复 splay+贪心
找平衡树练习题的时候发现了这道神题,可以说这道题是近几年单考splay的巅峰之作了. 题目大意:给出括号序列,实现区间翻转,区间反转和区间更改.查询区间最少要用几次才能改成合法序列. 分析: 首先我们 ...
- 【bzoj2329】[HNOI2011]括号修复 Splay
题目描述 题解 Splay 由于有区间反转操作,因此考虑Splay. 考虑答案:缩完括号序列后剩下的一定是 $a$ 个')'+ $b$ 个'(',容易发现答案等于 $\lceil\frac a2\rc ...
- BZOJ2329: [HNOI2011]括号修复(Splay)
解题思路: Replace.Swap.Invert都可以使用Splay完美解决(只需要解决一下标记冲突就好了). 最后只需要统计左右括号冲突就好了. 相当于动态统计最大前缀合和最小后缀和. 因为支持翻 ...
- BZOJ 2329: [HNOI2011]括号修复 [splay 括号]
题目描述 一个合法的括号序列是这样定义的: 空串是合法的. 如果字符串 S 是合法的,则(S)也是合法的. 如果字符串 A 和 B 是合法的,则 AB 也是合法的. 现在给你一个长度为 N 的由‘(' ...
- [HNOI2011][bzoj 2329] 括号修复 [splay+前缀和]
题面: http://www.lydsy.com/JudgeOnline/problem.php?id=2329 思路: 显然,操作4中输出补全的最小费用是关键 这决定了我们不可能在splay上只维护 ...
- BZOJ 2329/2209 [HNOI2011]括号修复 (splay)
题目大意: 让你维护一个括号序列,支持 1.区间修改为同一种括号 2.区间内所有括号都反转 3.翻转整个区间,括号的方向不变 4.查询把某段区间变为合法的括号序列,至少需要修改多少次括号 给跪了,足足 ...
- 【BZOJ2329/2209】[HNOI2011]括号修复/[Jsoi2011]括号序列 Splay
[BZOJ2329/2209][HNOI2011]括号修复/[Jsoi2011]括号序列 题解:我们的Splay每个节点维护如下东西:左边有多少多余的右括号,右边有多少多余的左括号,同时为了反转操作, ...
- 【BZOJ-2329&2209】括号修复&括号序列 Splay
2329: [HNOI2011]括号修复 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 1007 Solved: 476[Submit][Statu ...
随机推荐
- div不设置高度背景颜色或外边框不能显示的解决方法
在使用div+css进行网页布局时,如果外部div有背景颜色或者边框,而不设置其高度,在浏览时出现最外层Div的背景颜色和边框不起作用的问题. 大体结构<div class="oute ...
- 如何成为优秀评级卖家(Top-rated seller)?与超级卖家的区别是
以eBay美国站点为例,要成为优秀评级卖家(Top-rated seller),需满足如下条件: ● 先成为 eBay超级卖家 ● Low DSR (US buyers) <= 0.50% 或 ...
- Linux 发展史与vm安装linux centos 6.9
操作系统 是一个人与计算机硬件的中介. Linux操作系统 开源代码的.自由传播的类Unix操作系系统软件: 多用户.多任务.多线程.多CPU的操作系统. 服务器端.嵌入式开发.个人pc桌面,服务器领 ...
- Scrum立会报告+燃尽图(十月二十三日总第十四次)
此作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2246 项目地址:https://git.coding.net/zhang ...
- SQL连接的方法
1.创建连接字符串 string con = "Data Source=127.0.0.1;Initial Catalog=dingdan;Integrated Security=True& ...
- 按照Right-BICEP要求设计的测试用例
测试用例: 测试方法:Right-BICEP 测试要求: Right-结果是否正确? B-是否所有的边界条件都是正确的? P-是否满足性能要求? 题目是否有重复? 数量是否可定制? 数值范围是否可定制 ...
- 用P4对数据平面进行编程
引言 SDN架构强调了对控制平面的可编程,数据平面只负责转发,导致数据平面很大程度上受制于功能固定的包处理硬件. P4语言的特性: 目标无关性:P4语言不受制于具体设备,所有可编程芯片都可以使用P4编 ...
- c++团队作业工作笔记
这周时间还比较充裕,所以就有较多的时间来投入团队作业之中. emmmm,由于组长那边感觉完全没动,于是我完成了选英雄的UI界面,到时候给button加上信号就没什么问题. 虽然界面比较简单,但是还是花 ...
- 中国剩余定理---FZU 1402 猪的安家
J - 猪的安家 Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Submit Sta ...
- 【技术向】rainmeter的设计与发现
我们在大学期间所学的那点代码知识还远远不够,于是我就自己寻找到了一款简单易懂的软件,来丰富我的代码知识. 这款软件叫rainmeter,中文叫做雨滴,是一款可以修改桌面的软件.它可以将桌面上更改出硬盘 ...
