bzoj 2209: [Jsoi2011]括号序列 splay
2209: [Jsoi2011]括号序列
Time Limit: 20 Sec Memory Limit: 259 MB
Submit: 833 Solved: 392
[Submit][Status]
Description

Input
Output
Sample Input
)(())(
0 1 6
0 1 4
0 3 4
Sample Output
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的更多相关文章
- 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 ...
- ●BZOJ 2209 [Jsoi2011]括号序列
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=2209 题解: Splay 很好的题,但是把智障的我给恶心到了... 首先不难发现,最后没 ...
- 【BZOJ2329/2209】[HNOI2011]括号修复/[Jsoi2011]括号序列 Splay
[BZOJ2329/2209][HNOI2011]括号修复/[Jsoi2011]括号序列 题解:我们的Splay每个节点维护如下东西:左边有多少多余的右括号,右边有多少多余的左括号,同时为了反转操作, ...
- 【BZOJ】2209: [Jsoi2011]括号序列(splay)
http://www.lydsy.com/JudgeOnline/problem.php?id=2209 splay又犯逗........upd1那里的sum忘记赋值反............. 本题 ...
- BZOJ2209 [Jsoi2011]括号序列 splay
原文链接http://www.cnblogs.com/zhouzhendong/p/8093556.html 题目传送门 - BZOJ2209 题解 我太弱了,调出这题感觉都要吐了. 题解懒得写了. ...
- 【BZOJ-2329&2209】括号修复&括号序列 Splay
2329: [HNOI2011]括号修复 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 1007 Solved: 476[Submit][Statu ...
- BZOJ 2329/2209 [HNOI2011]括号修复 (splay)
题目大意: 让你维护一个括号序列,支持 1.区间修改为同一种括号 2.区间内所有括号都反转 3.翻转整个区间,括号的方向不变 4.查询把某段区间变为合法的括号序列,至少需要修改多少次括号 给跪了,足足 ...
- BZOJ 2329: [HNOI2011]括号修复( splay )
把括号序列后一定是))))((((这种形式的..所以维护一个最大前缀和l, 最大后缀和r就可以了..答案就是(l+1)/2+(r+1)/2...用splay维护,O(NlogN). 其实还是挺好写的, ...
随机推荐
- C# richTextBox编辑器
附件:http://files.cnblogs.com/xe2011/CSHARP_RichTextBoxEditor.rar 完整的转到这里 http://www.cnblogs.com/xe201 ...
- Java用链表实现栈和队列
1.用链表实现栈 package stack; /** * * @author denghb * */ class Link { public long dData; public Link next ...
- 让Tomcat支持中文路径名和中文文件名
http://hdwangyi.iteye.com/blog/107709 Tomcat是Java开发者使用得较多的一个Web服务器,因为它占用资源小,运行速度快等特点,深受Java Web程序员的喜 ...
- MFC 一个类訪问还有一个类成员对象的成员变量值
作者:卿笃军 原文地址:http://blog.csdn.net/qingdujun/article/details/35263857 MFC中一个类要訪问另外一个类的的对象的成员变量值,这就须要获得 ...
- careercup-数组和字符串1.5
1.5 利用字符重复出现的次数,编写一个方法,实现基本的字符串压缩功能.比如,字符串”aabcccccaaa“会变成”a2b1c5a3“.若”压缩“后的字符串没有变短,则返回原先的字符串. 类似 le ...
- python学习笔记--Django入门0 安装dangjo
经过这几天的折腾,经历了Django的各种报错,翻译的内容虽然不错,但是与实际的版本有差别,会出现各种奇葩的错误.现在终于找到了解决方法:查看英文原版内容:http://djangobook.com/ ...
- Java基础知识强化之IO流笔记29:BufferedOutputStream / BufferedInputStream(字节缓冲流)之BufferedInputStream读取数据
1. BufferedInputStream读取数据 BufferedInputStream构造方法,如下: 构造方法摘要 BufferedInputStream(InputStream in) ...
- JavaScript与html5写的贪吃蛇完整代码
JavaScript与html5写的贪吃蛇完整代码 查看运行效果可访问http://www.codesocang.com/texiao/youxitexiao/2014/0402/7045.html# ...
- 在Silverlight中使用async/await
现在 async/await 大行其道,确实,有了 async/await ,异步编程真是简单多了,个人觉得 async/await 的出现,给开发者还来的方便,绝不亚于当年 linq 的出现. 但要 ...
- UTF-8 BOM对PHP的影响
今天在用notepad++写代码时 载入一个frameset框架模版后 在页面上一直不显示该页面,查看源码后都正常.然后索性把里面东西全删掉 随便写了几个测试文字可以正常显示. 折腾了好长时间,最后偶 ...