BZOJ 3223 文艺平衡树 [codevs3303翻转区间]
AC通道:http://www.lydsy.com/JudgeOnline/problem.php?id=3223
通道2:http://codevs.cn/problem/3303/
题目分析:
我们先想想数组怎么操作。
每次O(1)找到区间位置,然后旋转,大概是O(r-l+1)的。
这样太慢了,每次我都将操作进行到底了。可是翻转标记有时候其实不需要每个都去办,因为它们之间还可以互相抵消。
能不能用线段树呢?
找到区间log(n),标记一下O(1),可以过?
不行!这个翻转显得与往常处理的线段树题目不一样,因为之前线段树是静态的数轴,但是这个数轴会翻转!
直接根据下标找到的只是数值,不是现在它在数组中的位置。每次怎么找到新的区间位置?
每次去找的时候,将标记下传然后翻转线段树[交换左右节点]好像就行了...
线段树真的可以翻转么?我每个节点的信息岂不是非常混乱!下次再想找到这个区间就不容易了,因为我线段树的数值不满足当初的性质了。
通过刚才的分析,发现这题中的数值其实没有多大作用,只是它特殊的一个标记而已,重要的还是它在序列中的位置,以及如何处理好翻转与查询的关系。
翻转是不能彻底进行的,要用标记且标记不能下传到底,但是在这样的条件下要有一种快速的算法,让我们找到我们需要的区间。
标记让人想到树结构,快速找区间又想到二叉树,于是不妨维护一颗搜索二叉树,其中以这个元素在序列中的位置作为排序的标准。
在每次寻找左右端点时,即使有一些节点的标记还未下传,但是只要这个节点下的子树的元素个数小于当前我的查询值,我就可以跳过它。
反之,就将它的标记下传,并让它的左右子树交换位置,依次往下,直到找到自己需要的那个节点上。
节点到区间的转变怎么实现呢?这边需要Splay()操作了。
首先将l-1转到根节点上来,然后将r+1转到l-1的右节点上。
这样[l,r]就很清晰而且全部存在于一棵子树下了。
有没有觉得这个算法的神奇之处啊?或是说Splay()也可以这样用,标记也可以这样用...Orz
当然这题还有几个地方需要注意,一是当l=1时,需要将排行第0的节点转上来(如果是n的话,你多虚拟一个n+1的节点是可以的)...不可实现啊,
【其实你也可以特判掉,在这种情况下只找r+1翻到顶上(对于n也可以这样特判)】,不过...笔者表示膜别人思路将所有节点的下标前移了一个,所以代码中表现的是将l和r+2进行操作。
二是刚开始建树的话,不仅要注意建到n+2【如果采用特判则是n+1】,也要尽量让它初始就平衡一点哦...
#include<cstdio>
#include<cstring>
#include<algorithm> using namespace std; inline int in(){
int x=;char ch=getchar();
while(ch>'' || ch<'') ch=getchar();
while(ch>='' && ch<='') x=x*+ch-'',ch=getchar();
return x;
} const int maxn=; int n,m,rt; struct Node{
int sz,f;
int l,r;
bool mk; void trans(){swap(l,r);}
}s[maxn]; inline void updata(int x){
s[x].sz=s[s[x].l].sz+s[s[x].r].sz+;
} int build(int l,int r){
if(l>r) return ;
int mid=(l+r)>>;
s[mid].l=build(l,mid-);
s[mid].r=build(mid+,r);
updata(mid);
s[s[mid].l].f=mid,s[s[mid].r].f=mid;
return mid;
} inline void push_down(int x){
if(s[x].mk){
s[x].trans();
s[s[x].l].mk^=;
s[s[x].r].mk^=;
s[x].mk=;
}
} void zig(int x){
int y=s[x].f; s[x].f=s[y].f;
if(s[y].f){
if(y==s[s[y].f].l) s[s[y].f].l=x;
else s[s[y].f].r=x;
} s[y].l=s[x].r;
if(s[x].r)
s[s[x].r].f=y; s[y].f=x,s[x].r=y; updata(y);updata(x);
} void zag(int x){
int y=s[x].f; s[x].f=s[y].f;
if(s[y].f){
if(y==s[s[y].f].l) s[s[y].f].l=x;
else s[s[y].f].r=x;
} s[y].r=s[x].l;
if(s[x].l)
s[s[x].l].f=y; s[x].l=y,s[y].f=x; updata(y);updata(x);
} void Splay(int x,int gf){
int y;
while(s[x].f!=gf){
y=s[x].f;
if(s[y].f==gf){
if(x==(s[y].l)) zig(x);
else zag(x);
}
else{
int z=s[y].f;
if(y==s[z].l){
if(x==s[y].l) zig(y),zig(x);
else zag(x),zig(x);
}
else{
if(x==s[y].r) zag(y),zag(x);
else zig(x),zag(x);
}
}
}
if(!gf) rt=x;
} int find(int k){
int p=rt;
if(k>s[p].sz) return ;
while(p){
push_down(p);//每访问到一个节点,就将它的标记下传,从这个点到根的路上就都传完了
if(k<=s[s[p].l].sz) p=s[p].l;
else{
k-=s[s[p].l].sz;
if(k==) return p;
k--,p=s[p].r;
}
}
} void rev(int l,int r){
int a=find(l),b=find(r+);//这里是找到这两个需要翻转的节点的位置
Splay(a,);Splay(b,a);
s[s[b].l].mk^=;
} void print(int x){
if(!x) return;
push_down(x);
print(s[x].l);
if(x> && x<=n+) //有几个节点是根据需要虚拟出的,它们不能输出
printf("%d ",x-);
print(s[x].r);
} int main(){
#ifndef ONLINE_JUDGE
freopen("3223.in","r",stdin);
freopen("3223.out","w",stdout);
#endif int l,r; n=in();m=in();
rt=build(,n+);
while(m--)
l=in(),r=in(),rev(l,r);
print(rt);
return ;
}
BZOJ 3223 文艺平衡树 [codevs3303翻转区间]的更多相关文章
- bzoj3223 文艺平衡树 codevs3303 翻转区间
splay模版题吧 只有区间翻转 至于为什么要把须翻转区间旋到根 因为查找一个区间可以先找出他左端点左边第一个点和右端点x右边第一个点y 然后将x旋到根节点 y旋到x的右儿子 这样x的右边的点就是所有 ...
- [题解]bzoj 3223 文艺平衡树
3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 3884 Solved: 2235[Submit][Sta ...
- bzoj 3223 文艺平衡树 - Splay
3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 3884 Solved: 2235[Submit][Sta ...
- BZOJ 3223 文艺平衡树
Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 ...
- bzoj 3223 文艺平衡树 splay 区间翻转
Tyvj 1728 普通平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 17715 Solved: 7769[Submit][Status][ ...
- bzoj 3223 文艺平衡树 Splay 打标志
是NOI2003Editor的一个子任务 #include <cstdio> #include <vector> #define maxn 100010 using names ...
- BZOJ 3223: Tyvj 1729 文艺平衡树-Splay树(区间翻转)模板题
3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 6881 Solved: 4213[Submit][Sta ...
- [BZOJ 3223 & Tyvj 1729]文艺平衡树 & [CodeVS 3243]区间翻转
题目不说了,就是区间翻转 传送门:BZOJ 3223 和 CodeVS 3243 第一道题中是1~n的区间翻转,而第二道题对于每个1~n还有一个附加值 实际上两道题的思路是一样的,第二题把值对应到位置 ...
- [bzoj3224]普通平衡树/3223文艺平衡树
这是一道很普通的题.. 最近花了很多时间来想要去干什么,感觉自己还是太拿衣服 做这道题是因为偶尔看到了lavender的blog和她的bzoj早期AC记录,就被题目深深地吸引到了,原因有二: 自己sp ...
随机推荐
- 【漫画解读】HDFS存储原理(转载)
以简洁易懂的漫画形式讲解HDFS存储机制与运行原理. 一.角色出演 如上图所示,HDFS存储相关角色与功能如下: Client:客户端,系统使用者,调用HDFS API操作文件;与NN交互获取文件元数 ...
- 一款jQuery实现重力弹动模拟效果特效,弹弹弹,弹走IE6
一款jQuery实现重力弹动模拟效果特效 鼠标经过两块黑色div中间的红色线时,下方的黑快会突然掉落, 并在掉落地上那一刻出现了弹跳的jquery特效效果,非常不错,还兼容所有的浏览器, 适用浏览器: ...
- C++判断五位以内的对称素数
题目内容:判断一个数是否为对称且不大于五位数的素数. 输入描述:输入数据含有不多于50个的正整数n(0<n<232). 输出描述:对于每个n,如果该数是不大于五位数的对称素数,则输出“Ye ...
- 17) JMS: java Message Service(Java消息服务)
JMS是一个标准,就像EJB,有很多开源的,商业的实现,ms技术对应的规范是jsr914,规范的实现称为jms provider,常见的实现有ActiveMQ.JBoss MQ.IBM We ...
- 写一个函数,尽可能高效的,从一个标准 url 里取出文件的扩展名
例如: http://www.sina.com.cn/abc/de/fg.php?id=1 需要取出 php 或 .php function getExt($url){ $arr=parse_url( ...
- devexpress 数据导入(gridcontrol 导出 csv)
// 1.gridcontrol 导出 csv: DataTable dtbNew = new DataTable(); dtbNew.Columns.Add().GetType()); dtbNew ...
- C# 调用系统API 内核 简单样例
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.R ...
- Python核心编程--学习笔记--4--Python对象
现在开始学习Python语言的核心部分.首先了解什么是Python对象,然后讨论最常用的内建类型,接下来讨论标准类型运算符和内建函数,之后给出对标准类型的不同分类方式,最后提一提Python目前还不支 ...
- 在newegg工作的这两个月
6月11号,接到录用通知后的第二天,来到了Newegg . 作为开发,在本职工作上 1.入职Quick Start: 两周多的入职快速指引,以了解业务,架构为目的. 因为之前一直有用思维导图的习惯,所 ...
- Map,HashMap
Map(映射),又称为字典(Dictionary),是由关键字(Key)及其对应的元素值(Value)所组成的元素单元(Element)的表单式集合. 通常,对于Map而言,使用给定的Key,可以迅速 ...