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). 其实还是挺好写的, ...
随机推荐
- android_handler(三)
这篇记录 android 消息机制中,MainThread 向 WorkThread 发送消息.( MainThread → WorkThread ) 步骤: 1.准备looper对象 2.在子线程中 ...
- 部分 CM11 系统 Android 平板执行植物大战僵尸 2 黑屏的解决的方法
原文 http://forum.xda-developers.com/showthread.php?t=2755197 部分 CM11 系统的 Android 平板(比如三星 GT-P5110 )执行 ...
- linux中创建静态库和动态库
1. 函数库有两种:静态库和动态库. 静态库在程序编译的时候会被连接到目标代码中,程序运行时将不再需要改静态库. 动态库中程序编译的时候并不会连接到目标代码中,而是在程序运行时才被载入,因此在程序运行 ...
- 如何查看电脑的链接WIFI密码
这个查看起来确实有难度,一般软件也难以破解.不如你试试在浏览器输入192.168.1.1,账户密码都是admin然后进入路由设置里.无线安全设置里有.
- Graphql graffiti
https://github.com/RisingStack/graffiti-mongoose https://blog.risingstack.com/graffiti-mongoose-mong ...
- Java基础知识强化之集合框架笔记51:Map集合之Map集合的功能概述与测试
1. Map集合的功能概述 (1)添加功能 V put(K key,V value):添加元素.这个其实还有另一个功能?先不告诉你,等会讲 如果键是第一次存储,就直接存储元素,返回null 如果键不是 ...
- 利用AWS简单存储服务(S3)托管网站
1.首先建立Storage Bucket存储桶,名为网站域名: 2.在[属性]中选择启用网站托管或重定向到另一主机,即可. 3.官方参考文档:https://docs.aws.amazon.com/z ...
- android Lib
Android 支持库软件包含可以添加至应用的多个库.每个库均支持特定范围的 Android 平台版本和功能. 本指南介绍了各支持库提供的重要功能和版本支持,从而帮助您决定在应用中添加哪些支持库.一般 ...
- PHP 正则通配符
$a = preg_match('/ph+p/','aaaphpbbbp'); +的前导就是h $a = preg_match('/ph+p/','aaaphhhhhhhhhhpbbbp'); //第 ...
- [FTP] FTPOperater--FTP操作帮助类 (转载)
点击下载 FTPOperater.zip 这个类是关于FTP的一些操作的1.连接FTP服务器 2.上传3.下载4.删除文件5.获取当前目录下明细(包含文件和文件夹) 6.获取FTP文件列表(包括文件 ...