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). 其实还是挺好写的, ...
随机推荐
- 无法定位序数XX于动态链接库XX.dll的解决的方法
问题阐述: 开发环境:VS2008 使用RELEASE生成了可执行文件,发如今某些电脑上能够正常执行,但在部分电脑中执行失败提示:无法定位序数8523于动态链接库mfc90.dll 在网上查找了一些资 ...
- myeclipes使用过程中的错误解决方案
1.‘Building workspace’ has encountered a problem. Errors occurred during the build. 解决方案:这样的错误,主要是由于 ...
- ActivityGroup相关--getLocalActivityManager()
ActivityGroup简介 1.ActivityGroup的核心就是继承了该类,能够通过getLocalActivityManager()得到一个LocalActivityManager 如,Lo ...
- JAVA向文件中追加内容(转)
向文件尾加入内容有多种方法,常见的方法有两种: RandomAccessFile类可以实现随机访问文件的功能,可以以读写方式打开文件夹的输出流 public void seek(long pos)可以 ...
- android 72 确定取消对话框,单选对话框,多选对话框
package com.itheima.dialog; import android.os.Bundle; import android.app.Activity; import android.ap ...
- wget下载网站整个目录
wget -r -p -np -k -P ./data/ http://example.com/eg/ 具体参数: -P 表示下载到哪个目录 -r 表示递归下载 -np 表示不下载旁站连接 -k 表示 ...
- JS如何封装一些列方法为一个对象的操作,然后集中管理这些操作,方便修改和调用
var Api = { ajax:{ // 添加项目 旧! add_project : function(pro_name, html, css, js,callback) { $.post(&quo ...
- Python开发实战教程(8)-向网页提交获取数据
来这里找志同道合的小伙伴!↑↑↑ Python应用现在如火如荼,应用范围很广.因其效率高开发迅速的优势,快速进入编程语言排行榜前几名.本系列文章致力于可以全面系统的介绍Python语言开发知识和相关知 ...
- jQuery实现多级手风琴树形下拉菜单(源码)
前几天因为公司的菜单要调整,公司的UI框架是不支持的,所以就自己在网上找了一个下拉菜单,可以支持多级菜单数据的,菜单数据是从xml文件中配置后读取的,网上有许多这方面的例子感觉不是很好用,就打了个包贴 ...
- js中eval函数
后台数据 // 回显复选框用 List<Long> tempRoles = new ArrayList<Long>(); @SuppressWarnings("unc ...