题目链接

Splay基础操作

\(Splay\)上的区间翻转

首先,这里的\(Splay\)维护的是一个序列的顺序,每个结点即为序列中的一个数,序列的顺序即为\(Splay\)的中序遍历

那么如何实现区间翻转呢?

对于一次区间翻转操作\(rev(l,r)\),显然先要找到\(l\)和\(r\)在\(Splay\)中的位置

然后把\(l-1\) \(splay\)到根结点,再把\(r+1\) \(splay\)到\(l\)的右儿子的位置

那么区间\([l,r]\)就到了一个子树上,即\(ch[r+1][0]\)

这样就可以翻转了

把序列上的\([l,r]\)翻转即为把\(Splay\)上\([l,r]\)的子树给翻过来

当然不是这么翻。。

没错!是这么翻。。好诡异

我们只需要把整棵子树的所有节点的左右儿子脚滑交换一下就可以了,可以维护一个类似于线段树区间操作的\(tag\)标记

只需要在查找和最后中序遍历时\(push\_down\)就可以了

完整代码:

#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std; const int MAXN=100010;
const int INF=0x3f3f3f3f; inline int read(){
int x=0;char c=getchar();
while(c<'0')c=getchar();
while(c>='0')x=(x<<1)+(x<<3)+c-'0',c=getchar();
return x;
} int n,m,root,Size;
int ch[MAXN][2],f[MAXN],size[MAXN],val[MAXN],tag[MAXN]; inline int get_w(int x){
return ch[f[x]][1]==x;
} inline void push_up(int x){
if(x)size[x]=size[ch[x][0]]+size[ch[x][1]]+1;
} inline void rotate(int x){
int fa=f[x],gfa=f[f[x]],ws=get_w(x);
ch[fa][ws]=ch[x][ws^1];f[ch[fa][ws]]=fa;
ch[x][ws^1]=fa;f[fa]=x;f[x]=gfa;
if(gfa)ch[gfa][ch[gfa][1]==fa]=x;
push_up(fa);push_up(x);
} inline void Splay(int x,int goal){
for(int fa;fa=f[x];rotate(x))
if(fa==goal)break;
else if(f[fa]!=goal) rotate(get_w(x)==get_w(fa)?fa:x);
if(goal==0)root=x;
} inline void insert(int x){
if(root==0){
root=++Size;val[Size]=x;
size[Size]=0;return;
}
int now=root,fa=0;
while(now)fa=now,now=ch[now][x>val[now]];
val[++Size]=x;
f[Size]=fa;ch[fa][x>val[fa]]=Size;
size[Size]=1;Splay(Size,0);
} inline void push_down(int p){ //标记下放
if(tag[p]){
swap(ch[p][0],ch[p][1]);
tag[ch[p][0]]^=1;
tag[ch[p][1]]^=1;
tag[p]=0;
}
} inline int find_num(int x){ //找到序列的第x个数的位置
if(!root)return 0;
int now=root;
while(1){
push_down(now);
if(x<=size[ch[now][0]])now=ch[now][0];
else{
int temp=size[ch[now][0]]+1;
if(x<=temp)return now;
else x-=temp,now=ch[now][1];
}
}
} void print(int x){ //输出答案(中序遍历一遍)
if(!x)return;
push_down(x);
print(ch[x][0]);
if(val[x]!=INF&&val[x]!=-INF)
printf("%d ",val[x]);
print(ch[x][1]);
} void reserve(int l,int r){ //翻转区间[l,r]
int x=find_num(l);
int y=find_num(r+2);
Splay(x,0);Splay(y,x);
tag[ch[ch[root][1]][0]]^=1;
} int main(){
n=read();m=read();
insert(-INF);insert(INF); //翻转[1,x]或[x,n]时需要用到0或n
for(int i=1;i<=n;++i)insert(i);
int l,r;
while(m--){
l=read();r=read();
reserve(l,r);
}
print(root);
puts("");
return 0;
}

【洛谷P3391】文艺平衡树——Splay学习笔记(二)的更多相关文章

  1. [洛谷P3391] 文艺平衡树 (Splay模板)

    初识splay 学splay有一段时间了,一直没写...... 本题是splay模板题,维护一个1~n的序列,支持区间翻转(比如1 2 3 4 5 6变成1 2 3 6 5 4),最后输出结果序列. ...

  2. 洛谷P3391文艺平衡树(Splay)

    题目传送门 转载自https://www.cnblogs.com/yousiki/p/6147455.html,转载请注明出处 经典引文 空间效率:O(n) 时间效率:O(log n)插入.查找.删除 ...

  3. BZOJ3223/洛谷P3391 - 文艺平衡树

    BZOJ链接 洛谷链接 题意 模板题啦~2 代码 //文艺平衡树 #include <cstdio> #include <algorithm> using namespace ...

  4. BZOJ3224/洛谷P3391 - 普通平衡树(Splay)

    BZOJ链接 洛谷链接 题意简述 模板题啦~ 代码 //普通平衡树(Splay) #include <cstdio> int const N=1e5+10; int rt,ndCnt; i ...

  5. 洛谷 P3391 文艺平衡树

    题目描述 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 --b ...

  6. 洛谷P3391 文艺平衡树 (Splay模板)

    模板题. 注意标记即可,另外,涉及区间翻转操作,记得设立首尾哨兵. 1 #include<bits/stdc++.h> 2 using namespace std; 3 const int ...

  7. 文艺平衡树 Splay 学习笔记(1)

    (这里是Splay基础操作,reserve什么的会在下一篇里面讲) 好久之前就说要学Splay了,结果苟到现在才学习. 可能是最近良心发现自己实在太弱了,听数学又听不懂只好多学点不要脑子的数据结构. ...

  8. 平衡树splay学习笔记#1

    这一篇博客只讲splay的前一部分的操作(rotate和splay),后面的一段博客咕咕一段时间 后一半的博客地址:[传送门] 前言骚话 为了学lct我也是拼了,看了十几篇博客,学了将近有一周,才A掉 ...

  9. 平衡树splay学习笔记#2

    讲一下另外的所有操作(指的是普通平衡树中的其他操作) 前一篇的学习笔记连接:[传送门],结尾会带上完整的代码. 操作1,pushup操作 之前学习过线段树,都知道子节点的信息需要更新到父亲节点上. 因 ...

随机推荐

  1. 2、Shell命令学习笔记

    1.Shell命令行解释器 1.1 Shell命令解释器 Shell是一个特殊的应用程序,介于操作系统内核和用户之间,负责接收用户输入的操作指令(命令)并进行解释,将需要执行的操作传递给内核执行. 因 ...

  2. Nginx fastcgi_cache权威指南

    一.简介 Nginx版本从0.7.48开始,支持了类似Squid的缓存功能.这个缓存是把URL及相关组合当做Key,用Md5算法对Key进行哈希,得到硬盘上对应的哈希目录路径,从而将缓存内容保存在该目 ...

  3. selenium自学笔记---下拉框定位元素select

    下拉框1.先定位select 然后在定位option city = driver.find_element_by_id("selCities_0") city.find_eleme ...

  4. 转:更改pip源至国内镜像,显著提升下载速度

    经常在使用python的时候需要安装各种模块,而pip是很强大的模块安装工具,但是由于国外官方pypi经常被墙,导致不可用,所以我们最好是将自己使用的pip源更换一下,这样就能解决被墙导致的装不上库的 ...

  5. Mysql中类似于oracle中nvl()函数的ifnull()函数

    IFNULL(expr1,expr2)  如果expr1不是NULL,IFNULL()返回expr1,否则它返回expr2.IFNULL()返回一个数字或字符串值,取决于它被使用的上下文环境.  my ...

  6. 笔谈AudioToolbox(一)

    “五一”长假过的真快,三天就这么过去了.新的一周开始了,这周搞搞iOS平台上音频的解码与播放.动手咯,切入AudioToolbox.framework的学习,这个库太强大了,要想彻底弄懂不简单,从某种 ...

  7. C#操作mongodb(聚合函数)-分组找出每组的最大值

    public static void OnQuery_QXData(string DBName, string tablename,string layername)        {         ...

  8. MySQL DDL--gh-ost学习

    gh-ost工作原理 1.首先新建一张ghost表,结构与源表相同 2.使用alter命令修改ghost表 3.1.模拟从库命令获取主库上该表的binlog(基于全镜像的行模式的binlog包含更改前 ...

  9. oracle 11g goldengate搭建(一)

    初学ogg,基本了解ogg原理及架构之后,趁热打铁,搭建一个简单的学习环境,以实现目标:将sourcedb数据库的2个表:sourceuser.test01和sourceuser.dept通过ogg分 ...

  10. MySQL-CentOS7上安装Mysql5.7

    #安装 wget http://dev.mysql.com/get/mysql57-community-release-el7-11.noarch.rpm .noarch.rpm yum instal ...