[HNOI2011]括号修复 / [JSOI2011]括号序列
传送门
Solution
一道题花费了两天的时间……
在大佬
@PinkRabbit的帮助下,终于AC了,感动……首先,我们考虑一个括号序列被修改成合法序列需要的次数:
- 我们需要修改的其实是形如
...)))))(((((...- 我们把
(看成是-1,把)看成是1,那么其实只要知道了区间的前缀最大值pr和后缀最小值sum-pr那么就有$$ans=\left \lceil \frac{pr}{2} \right \rceil+\left \lceil \frac{-sum+su}{2} \right \rceil$$
如果我们维护了区间的前缀最大值
pr和后缀最大值su和区间和sum
- 区间翻转:
swap(pr,su)- 区间反转:
pr=-(sum-su)同时su=-(sum-pr),这里的(sum-su)就是前缀最小值,因为区间取了反,所以就是前缀最大值啦- 区间覆盖:直接打标记
所有区间操作都可以用平衡树来实现
要注意:区间覆盖的优先级应大于另外两个操作
Code
#include<bits/stdc++.h>
#define ll long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
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<<3)+(x<<1)+ch-'0';ch=getchar();}
return x*f;
}
#define MN 100005
char s[MN];
class fhq
{
private:
int sz;
int pri[MN],ls[MN],rs[MN],siz[MN];
int pr[MN],su[MN],tt[MN],cover[MN],V[MN];
bool rev[MN],inv[MN];
inline unsigned int random()
{
static unsigned int x=23333;
return x^=x<<13,x^=x>>17,x^=x<<5;
}
inline void up(int x)
{
siz[x]=1+siz[ls[x]]+siz[rs[x]];
tt[x]=tt[ls[x]]+tt[rs[x]]+V[x];
pr[x]=max(pr[ls[x]],tt[ls[x]]+V[x]+pr[rs[x]]);
su[x]=max(su[rs[x]],tt[rs[x]]+V[x]+su[ls[x]]);
}
inline void update(int x,int opt)
{
if(!x) return;
if(opt==1)
{
std::swap(pr[x],su[x]);std::swap(ls[x],rs[x]);
rev[ls[x]]^=1;rev[rs[x]]^=1;
}
if(opt==2)
{
V[x]=-V[x];
pr[x]=-(tt[x]-pr[x]);su[x]=-(tt[x]-su[x]);
std::swap(pr[x],su[x]);
tt[x]=-tt[x];
inv[ls[x]]^=1;inv[rs[x]]^=1;
}
if(opt==3)
{
V[x]=cover[x];tt[x]=cover[x]*siz[x];
su[x]=max(0,tt[x]);pr[x]=max(0,tt[x]);
cover[ls[x]]=cover[rs[x]]=cover[x];
rev[ls[x]]=inv[ls[x]]=0;
rev[rs[x]]=inv[rs[x]]=0;
}
}
inline void down(int x)
{
if(cover[x]) update(ls[x],3),update(rs[x],3),cover[x]=0;
if(rev[x]) update(ls[x],1),update(rs[x],1),rev[x]=0;
if(inv[x]) update(ls[x],2),update(rs[x],2),inv[x]=0;
}
public:
int rt;
int Merge(int rt1,int rt2)
{
if(!rt1||!rt2) return rt2|rt1;
if(pri[rt1]<pri[rt2])
{
down(rt1),rs[rt1]=Merge(rs[rt1],rt2);
up(rt1);return rt1;
}
else
{
down(rt2),ls[rt2]=Merge(rt1,ls[rt2]);
up(rt2);return rt2;
}
}
void Split(int x,int k,int&rt1,int&rt2)
{
if(!x) return (void)(rt1=rt2=0);
down(x);
if(k<=siz[ls[x]])
{
Split(ls[x],k,rt1,rt2);
ls[x]=rt2;up(x);rt2=x;
}
else
{
Split(rs[x],k-siz[ls[x]]-1,rt1,rt2);
rs[x]=rt1;up(x);rt1=x;
}
}
void Build(int &x,int l,int r)
{
if(l>r) return;int mid=l+r>>1;
x=++sz;
V[x]=tt[x]=(s[mid]==')'?1:-1),pri[x]=random();
if(l==r) return(void)(pr[x]=su[x]=max(0,V[x]),siz[x]=1);
Build(ls[x],l,mid-1),Build(rs[x],mid+1,r);up(x);
}
void Reverse(int l,int r)
{
register int rt1,rt2,rt3,rt4;
Split(rt,l-1,rt1,rt2);Split(rt2,r-l+1,rt3,rt4);
rev[rt3]^=1;update(rt3,1);rt=Merge(rt1,Merge(rt3,rt4));
}
void Invert(int l,int r)
{
register int rt1,rt2,rt3,rt4;
Split(rt,l-1,rt1,rt2);Split(rt2,r-l+1,rt3,rt4);
inv[rt3]^=1;update(rt3,2);rt=Merge(rt1,Merge(rt3,rt4));
}
void Replace(int l,int r,int c)
{
register int rt1,rt2,rt3,rt4;
Split(rt,l-1,rt1,rt2);Split(rt2,r-l+1,rt3,rt4);
cover[rt3]=c,update(rt3,3);rev[rt3]=inv[rt3]=0;rt=Merge(rt1,Merge(rt3,rt4));
}
int Query(int l,int r)
{
register int rt1,rt2,rt3,rt4,ret;
Split(rt,l-1,rt1,rt2);
Split(rt2,r-l+1,rt3,rt4);
ret=(pr[rt3]+1)/2+(-tt[rt3]+pr[rt3]+1)/2;
rt=Merge(rt1,Merge(rt3,rt4));
return ret;
}
}T;
int main(){
register int n,m,l,r;
n=read(),m=read();
scanf("%s",s+1);T.Build(T.rt,1,n);
register char opt[10],c[2];
while(m--)
{
scanf("%s",opt+1);l=read(),r=read();
if(opt[1]=='R') scanf("%s",c),T.Replace(l,r,c[0]=='('?-1:1);
if(opt[1]=='S') T.Reverse(l,r);
if(opt[1]=='I') T.Invert(l,r);
if(opt[1]=='Q') printf("%d\n",T.Query(l,r));
}
return 0;
}
Blog来自PaperCloud,未经允许,请勿转载,TKS!
[HNOI2011]括号修复 / [JSOI2011]括号序列的更多相关文章
- 【BZOJ2329/2209】[HNOI2011]括号修复/[Jsoi2011]括号序列 Splay
[BZOJ2329/2209][HNOI2011]括号修复/[Jsoi2011]括号序列 题解:我们的Splay每个节点维护如下东西:左边有多少多余的右括号,右边有多少多余的左括号,同时为了反转操作, ...
- 洛谷 P3215 [HNOI2011]括号修复 / [JSOI2011]括号序列(fhq-treap)
题目链接 题意:有一个长度为 \(n\) 的括号序列,你需要支持以下操作: 将 \([l,r]\) 中所有括号变为 \(c\) 将 \([l,r]\) 区间翻转 将 \([l,r]\) 区间中左括号变 ...
- BZOJ 2329: [HNOI2011]括号修复 [splay 括号]
题目描述 一个合法的括号序列是这样定义的: 空串是合法的. 如果字符串 S 是合法的,则(S)也是合法的. 如果字符串 A 和 B 是合法的,则 AB 也是合法的. 现在给你一个长度为 N 的由‘(' ...
- BZOJ 2329: [HNOI2011]括号修复( splay )
把括号序列后一定是))))((((这种形式的..所以维护一个最大前缀和l, 最大后缀和r就可以了..答案就是(l+1)/2+(r+1)/2...用splay维护,O(NlogN). 其实还是挺好写的, ...
- bzoj千题计划222:bzoj2329: [HNOI2011]括号修复(fhq treap)
http://www.lydsy.com/JudgeOnline/problem.php?id=2329 需要改变的括号序列一定长这样 :)))((( 最少改变次数= 多余的‘)’/2 [上取整] + ...
- 【BZOJ-2329&2209】括号修复&括号序列 Splay
2329: [HNOI2011]括号修复 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 1007 Solved: 476[Submit][Statu ...
- bzoj 2209: [Jsoi2011]括号序列 splay
2209: [Jsoi2011]括号序列 Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 833 Solved: 392[Submit][Status ...
- BZOJ 2209: [Jsoi2011]括号序列 [splay 括号]
2209: [Jsoi2011]括号序列 Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 1111 Solved: 541[Submit][Statu ...
- bzoj 2209 [Jsoi2011]括号序列 平衡树
2209: [Jsoi2011]括号序列 Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 1404 Solved: 699[Submit][Statu ...
随机推荐
- 扩展 MongoDB.Driver 支持实体
针对MongoDB的官方C#驱动进行扩展 一.安装 Install-Package Apteryx.MongoDB.Driver.Extend 移步我的项目https://github.com/cod ...
- VBA分别使用MSXML的DOM属性和XPATH进行网页爬虫
本文要重点介绍的是VBA中的XmlHttp对象(MSXML2.XMLHTTP或MSXML.XMLHTTP),它可以向http服务器发送请求并使用微软XML文档对象模型Microsoft XML Doc ...
- MySQL中You can't specify target table '表名'('sn_app_label') for update in FROM clause错误解决办法
在有些时候有级联关系的数据放在了同一张表中,在写sql语句的时候可能会遇到这样的场景:我要插入两条数据,第一条是父节点,第二条是子节点,关联关系是父节点的自增长id:在写这样的sql语句时有可能就会出 ...
- Python爬虫之BeautifulSoap的用法
1. Beautiful Soup的简介 简单来说,Beautiful Soup是python的一个库,最主要的功能是从网页抓取数据.官方解释如下: Beautiful Soup提供一些简单的.pyt ...
- echarts使用结合时间轴timeline动态刷新案例
1.echarts简介 ECharts,一个使用 JavaScript 实现的开源可视化库,可以流畅的运行在 PC 和移动设备上,兼容当前绝大部分浏览器(IE8/9/10/11,Chrome,Fire ...
- RZ70注册SLD
本文的将S4 abap系统向PO JAVA系统中注册. S4 QASERPAP01 NR=60 ASCS=61 PO QASPISAP01 NR=60 SCS=61 http://qaspisap01 ...
- 表格分页——tablePagination
背景:表格是最为通用的展示方式,为了展示的统一性,以及分页组件的重用,这里写一个分页组件,供比较多或者较少数据2种表格进行分页展示. 分页组件: <template> <el-pag ...
- Android笔记(七) Android中的布局——线性布局
我们的软件是由好多个界面组成的,而每个界面又由N多个控件组成,Android中借助布局来让各个空间有条不紊的摆放在界面上. 可以把布局看作是一个可以放置很多控件的容器,它可以按照一定的规律调整控件的位 ...
- Linux安装java环境和maven
安装OpenJDK软件包: apt-get install openjdk-8-jdk 查看版本信息java -version 则代表安装成功 安装maven可以使用自己本机下载好的mavan使用Xf ...
- 管理Linux软件——apt
参考:Ubuntu的apt命令详解 apt命令是一个功能强大的命令行工具,它与Ubuntu的高级打包工具(APT,Advanced Packaging Tool )配合使用,可以执行安装新软件包,升级 ...