2209: [Jsoi2011]括号序列

Time Limit: 20 Sec  Memory Limit: 259 MB
Submit: 833  Solved: 392
[Submit][Status]

Description

Input

输入数据的第一行包含两个整数N和Q,分别表示括号序列的长度,以及操作的个数。 第二行包含一个长度为N的括号序列。 接下来Q行,每行三个整数t、x和y,分别表示操作的类型、操作的开始位置和操作的结 束位置,输入数据保证x不小于y。其中t=0表示询问操作、t=1表示反转操作、t=2表示翻转操 作。

Output

对于每一个询问操作,输出一行,表示将括号序列的该子序列修改为配对,所需的最少改动 个数。

Sample Input

6 3
)(())(
0 1 6
0 1 4
0 3 4

Sample Output

2
2
0

HINT

100%的数据满足N,Q不超过10^5

  终于算是会写splay了,这道题涉及到splay的区间翻转,取反,询问以及lazy标记下放等操作,算是涵盖了splay的基本用法。

  合法的括号序列的一个性质是,以正括号为1,反括号为-1,合法序列和为0,且所有前缀权值和非负。顾可以通过类似于线段树求区间最大子段和方式维护,由于涉及到区间翻转,故改为splay维护。

  这次编写问题还是在标记同步上,具体来说,在get_kth()调用前一定要down(),修改子树后要讲修改后的值传回根节点。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define MAXN 210000
#define MAXT 210000
inline int min(int x,int y,int z)
{
return min(x,min(y,z));
}
inline int min(int a,int b,int c,int d)
{
return min(min(a,b),min(c,d));
}
inline int max(int x,int y,int z)
{
return max(x,max(y,z));
}
inline int max(int a,int b,int c,int d)
{
return max(max(a,b),max(c,d));
}
int num[MAXN];
struct splay_tree
{
int ch[MAXT][],pnt[MAXT];
int val[MAXT],siz[MAXT],sum[MAXT],lx[MAXT][],rx[MAXT][];
bool rev[MAXT],neg[MAXT];
int stack[MAXT],tops;
int topt;
int root;
splay_tree()
{
topt=,root=;
tops=-;
}
void reverse(int now)
{
if (!now)return;
swap(lx[now][],rx[now][]);
swap(lx[now][],rx[now][]);
swap(ch[now][],ch[now][]);
rev[now]^=;
}
void negate(int now)
{
if (!now)return ;
val[now]=-val[now];
sum[now]=-sum[now];
swap(lx[now][],lx[now][]);
swap(rx[now][],rx[now][]);
lx[now][]=-lx[now][];
rx[now][]=-rx[now][];
lx[now][]=-lx[now][];
rx[now][]=-rx[now][];
neg[now]^=;
}
void update(int now)
{
lx[now][]=min(lx[ch[now][]][],sum[ch[now][]],sum[ch[now][]]+val[now],sum[ch[now][]]+val[now]+lx[ch[now][]][]);
lx[now][]=max(lx[ch[now][]][],sum[ch[now][]],sum[ch[now][]]+val[now],sum[ch[now][]]+val[now]+lx[ch[now][]][]);
rx[now][]=min(rx[ch[now][]][],sum[ch[now][]],sum[ch[now][]]+val[now],sum[ch[now][]]+val[now]+rx[ch[now][]][]);
rx[now][]=max(rx[ch[now][]][],sum[ch[now][]],sum[ch[now][]]+val[now],sum[ch[now][]]+val[now]+rx[ch[now][]][]);
sum[now]=sum[ch[now][]]+sum[ch[now][]]+val[now];
siz[now]=siz[ch[now][]]+siz[ch[now][]]+;
}
void down(int now)
{
if (rev[now])
{
reverse(ch[now][]);
reverse(ch[now][]);
rev[now]=;
}
if (neg[now])
{
negate(ch[now][]);
negate(ch[now][]);
neg[now]=;
}
}
void rotate(int now)
{
int p=pnt[now],anc=pnt[p];
int dir=ch[p][]==now;
if (anc)
ch[anc][ch[anc][]==p]=now;
pnt[now]=anc;
pnt[ch[now][dir]]=p;
ch[p][-dir]=ch[now][dir];
pnt[p]=now;
ch[now][dir]=p;
update(p);
update(now);
}
void splay(int now,int tp=)
{
int x=now;
tops=-;
while (x!=tp)
{
stack[++tops]=x;
x=pnt[x];
}
while (~tops)
{
down(stack[tops--]);
}
while (now!=tp && pnt[now]!=tp)
{
int p=pnt[now],anc=pnt[p];
if (anc==tp)
rotate(now);
else if ((ch[p][]==now) == (ch[anc][]==p))
rotate(p),rotate(now);
else
rotate(now),rotate(now);
}
if (tp==)
root=now;
}
int get_kth(int now,int rk)
{
if (!now)throw ;
down(now);
if (siz[ch[now][]]+==rk)
return now;
if (siz[ch[now][]]+<rk)
return get_kth(ch[now][],rk-siz[ch[now][]]-);
else
return get_kth(ch[now][],rk);
}
void insert(int pos,int v)
{
if (!root)
{
root=++topt;
pnt[topt]=;
val[topt]=v;
update(topt);
}else if (!pos)
{
splay(get_kth(root,pos));
ch[root][]=topt;
pnt[topt]=root;
val[topt]=v;
update(topt);
update(root);
}else
{
splay(get_kth(root,pos));
val[++topt]=v;
pnt[topt]=root;
ch[topt][]=ch[root][];
pnt[ch[root][]]=topt;
ch[root][]=topt;
update(topt);
update(root);
}
}
int get_result(int now)
{
int res=,t;
t=-lx[now][];
res=t/+t%;
t=res*+sum[now];
res+=abs(t)/;
return res;
}
int query(int l,int r)
{
if (l== && r==siz[root])
return get_result(root);
if (l==)
{
splay(get_kth(root,r+));
return get_result(ch[root][]);
}
if (r==siz[root])
{
splay(get_kth(root,l-));
return get_result(ch[root][]);
}
splay(get_kth(root,r+));
splay(get_kth(root,l-),root);
return get_result(ch[ch[root][]][]);
}
void make_reverse(int l,int r)
{
if (l== && r==siz[root])
return reverse(root);
if (l==)
{
splay(get_kth(root,r+));
reverse(ch[root][]);
update(root);
return ;
}
if (r==siz[root])
{
splay(get_kth(root,l-));
reverse(ch[root][]);
update(root);
return ;
}
splay(get_kth(root,r+));
splay(get_kth(root,l-),root);
reverse(ch[ch[root][]][]);
update(ch[root][]);
update(root);
}
void make_negate(int l,int r)
{
if (l== && r==siz[root])
return negate(root);
if (l==)
{
splay(get_kth(root,r+));
negate(ch[root][]);
update(root);
return ;
}
if (r==siz[root])
{
splay(get_kth(root,l-));
negate(ch[root][]);
update(root);
return ;
}
splay(get_kth(root,r+));
splay(get_kth(root,l-),root);
negate(ch[ch[root][]][]);
update(ch[root][]);
update(root); }
void scan(int now)
{
if (!now)return ;
if (pnt[ch[now][]]!=now && ch[now][])throw ;
if (pnt[ch[now][]]!=now && ch[now][])throw ;
if (siz[now]!=siz[ch[now][]]+siz[ch[now][]]+)throw ;
scan(ch[now][]);
printf("%d ",val[now]);
scan(ch[now][]);
}
}pp;
int main()
{
//freopen("input.txt","r",stdin);
// freopen("output.txt","w",stdout);
int n,m,i,j,k,x,y,z;
scanf("%d%d\n",&n,&m);
char ch;
for (i=;i<=n;i++)
{
scanf("%c",&ch);
if (ch==')')
pp.insert(i-,-);
else
pp.insert(i-,);
}
int opt;
for (i=;i<=m;i++)
{
scanf("%d%d%d",&opt,&x,&y);
// pp.scan(pp.root);printf("\n");
if (opt==)
{
printf("%d\n",pp.query(x,y));
}else if (opt==)
{
pp.make_negate(x,y);
}else if (opt==)
{
pp.make_reverse(x,y);
}
}
}

bzoj 2209: [Jsoi2011]括号序列 splay的更多相关文章

  1. BZOJ 2209: [Jsoi2011]括号序列 [splay 括号]

    2209: [Jsoi2011]括号序列 Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 1111  Solved: 541[Submit][Statu ...

  2. bzoj 2209 [Jsoi2011]括号序列 平衡树

    2209: [Jsoi2011]括号序列 Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 1404  Solved: 699[Submit][Statu ...

  3. ●BZOJ 2209 [Jsoi2011]括号序列

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=2209 题解: Splay 很好的题,但是把智障的我给恶心到了...   首先不难发现,最后没 ...

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

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

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

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

  6. BZOJ2209 [Jsoi2011]括号序列 splay

    原文链接http://www.cnblogs.com/zhouzhendong/p/8093556.html 题目传送门 - BZOJ2209 题解 我太弱了,调出这题感觉都要吐了. 题解懒得写了. ...

  7. 【BZOJ-2329&2209】括号修复&括号序列 Splay

    2329: [HNOI2011]括号修复 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 1007  Solved: 476[Submit][Statu ...

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

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

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

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

随机推荐

  1. SharePoint 2013 error The given assembly name or codebase System.ServiceModel.dll was invalid

    笔者近期在 SharePoint 2013 的环境中遇到一个奇怪的问题,前一天 SharePoint 2013 站点还是好好的.可是突然站点就报page can't display 500 错误: T ...

  2. CentOS下MySQL 5.7编译安装

    CentOS下MySQL 5.7编译安装   文章目录 安装依赖包 下载相应源码包 添加mysql用户 预编译 编译安装 启动脚本,设置开机自启动 /etc/my.cnf,仅供参考 初始化数据库 设置 ...

  3. Linux内核源代码情景分析系列

    http://blog.sina.com.cn/s/blog_6b94d5680101vfqv.html Linux内核源代码情景分析---第五章 文件系统  5.1 概述 构成一个操作系统最重要的就 ...

  4. 在String中添加移动构造函数和移动赋值运算符

    13.50 没有定义析构函数 #include<iostream> #include<string> #include<memory> #include<ut ...

  5. CentOS6.4下使用默认的文档查看器打开PDF文档乱码的解决方案

     最近在CentOS6.4下使用其默认的文档查看器打开PDF文档时出现乱码的方块,有两种方法可以解决.    方法一:修改/etc/fonts/conf.d/49-sansserif.conf文件,如 ...

  6. 节点类(CCNode)

    节点与渲染树 回顾前面的介绍,我们已经知道了精灵.层和场景如何构成一个游戏的框架.精灵属于层,层属于场景,玩家与精灵互动,并导致游戏画面在不同场景中切换.把每个环节拼接在一起,我们得到了一个完整的关系 ...

  7. 浅谈Android自定义锁屏页的发车姿势

    一.为什么需要自定义锁屏页 锁屏作为一种黑白屏时代就存在的手机功能,至今仍发挥着巨大作用,特别是触屏时代的到来,锁屏的功用被发挥到了极致.多少人曾经在无聊的时候每隔几分钟划开锁屏再关上,孜孜不倦,其酸 ...

  8. 常用JDBC连接字符串

    1.MySQL Class.forName( " org.gjt.mm.mysql.Driver " ); Connection conn = DriverManager.getC ...

  9. sqlite数据库修改及升级

    今天是上班的第二天,听说我最近的任务就是改bug,唉,权当学习了,遇到的一些问题都记录下来. sqlite数据库是android中非常常用的数据库,今天帮别人改bug,遇到一些问题记录下来. 1.修改 ...

  10. tp集成支付宝担保支付

    现在的网站功能越来越全乎了,很多网站都需要做支付功能,而且很多大平台都提供了各式各样的api来扩充自己的用户和开发者.话说,这种使用大平台的api来做支付,无论是从成本上还是从开发效率上都是很好的选择 ...