splay区间翻转
题目背景
这是一道经典的Splay模板题——文艺平衡树。 题目描述
您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 ,翻转区间是[,]的话,结果是5 输入输出格式
输入格式: 第一行为n,m n表示初始序列有n个数,这个序列依次是(,,⋯n−,n) m表示翻转操作次数 接下来m行每行两个数 [l,r] 数据保证1≤l≤r≤n 输出格式: 输出一行n个数字,表示原始序列经过m次变换后的结果 输入输出样例
输入样例#: 复制 输出样例#: 复制 说明
n,m≤
查看题面
题解
这道题目要求支持区间翻转,这是splay的典型操作。
基本思想:
- 只要将一棵BST每一个节点的左右子树交换,就相当于翻转次树的中序遍历
实现方法:
1. 用节点中序遍历的序号表示当前节点的下标,用key存节点的序号,经过一系列操作后中序遍历即为所求
2. 每次翻转区间【L,R】时只要将排名为L-1的点splay到根,将排名R+1的点splay到根的右节点,根的右子树的左子树的中序遍历就是【L,R】
如何将一个点splay到指定位置?
只要引入y标记,当前节点的父亲为y时跳出,就可以将x刚好splay到y的儿子
inline void splay(int x,int y){
for(int fa;(fa=f[x])!=y;rotate(x)){
if(f[fa]!=y){
rotate(get(fa)==get(x)?fa:x);
}
}
if(y==){
root=x;
}
}
1.标记是在每一次访问到一个新的节点是就要pushdown的(改变树的结构会破坏标记区间,所以先一步下传标记)
inline void push_down(int x){
if(x&&tag[x]){
tag[son[x][]]^=;
tag[son[x][]]^=;
swap(son[x][],son[x][]);
tag[x]=;
}
}
2.区分一个节点的排名和这个节点的值:这个节点的排名是它是当前数组中的第几个,用左儿子的size+1表示;这个节点的值是题目中输入的数字,在本题中是1~N
3.增加数字为1和N+2的两个哨兵节点+INF和-INF,因此第i个节点的下标为i+1。
4.难道交换左右子树不会破坏BST的性质吗?这就是容易混淆的一点,我们的区间操作是根据下标翻转的,用数组实现时下标就是数组地址,子树交换时,只是存储内容的改变,下标位置(树的形状)只会在旋转时改变,保证BST性质。
5.此题要求一次性插入N个节点,所以可以用线段树建树的方法O(N)建树
int init(int l,int r,int fa){
if(l>r){
return ;
}
int mid=(l+r)>>,cur=++sz;
key[cur]=val[mid];
f[cur]=fa;
son[cur][]=init(l,mid-,cur);
son[cur][]=init(mid+,r,cur);
upd(cur);
return cur;
}
完整代码
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int INF=1e9+,MAXN=1e5+,MAXM=1e5+;
int N,M,root,sz;
int key[MAXN],val[MAXN],f[MAXN],son[MAXN][],tag[MAXN],siz[MAXN];
inline void DEBUG(){
printf("root=%d siz=%d\n",root,sz);
for(int i=;i<=sz;i++){
printf("(idx=%d\tkey=%d\tlc=%d\trc=%d)\n",i,key[i],son[i][],son[i][]);
}
puts("-------------------------------------------------------");
}
inline void upd(int x){
if(x){
siz[x]=;
if(son[x][]){
siz[x]+=siz[son[x][]];
}
if(son[x][]){
siz[x]+=siz[son[x][]];
}
}
}
inline void push_down(int x){
if(x&&tag[x]){
tag[son[x][]]^=;
tag[son[x][]]^=;
swap(son[x][],son[x][]);
tag[x]=;
}
}
inline int get(int x){
return x==son[f[x]][];
}
inline void rotate(int x){
int fa=f[x],gf=f[fa],which=get(x);
son[fa][which]=son[x][which^];
f[son[fa][which]]=fa;
son[x][which^]=fa;
f[fa]=x;
f[x]=gf;
if(gf){
son[gf][son[gf][]==fa]=x;
}
upd(fa);
upd(x);
}
inline void splay(int x,int y){
for(int fa;(fa=f[x])!=y;rotate(x)){
if(f[fa]!=y){
rotate(get(fa)==get(x)?fa:x);
}
}
if(y==){
root=x;
}
}
int init(int l,int r,int fa){
if(l>r){
return ;
}
int mid=(l+r)>>,cur=++sz;
key[cur]=val[mid];
f[cur]=fa;
son[cur][]=init(l,mid-,cur);
son[cur][]=init(mid+,r,cur);
upd(cur);
return cur;
}
inline int findx(int x){//return the index
int cur=root;
while(){
push_down(cur);
if(son[cur][]&&x<=siz[son[cur][]]){
cur=son[cur][];
}else{
x-=siz[son[cur][]]+;
if(!x){
return cur;
}
cur=son[cur][];
}
}
}
void dfs(int x){
push_down(x);
if(son[x][]){
dfs(son[x][]);
}
if(key[x]!=INF&&key[x]!=-INF){
printf("%d ",key[x]);
}
if(son[x][]){
dfs(son[x][]);
}
}
int main(){
scanf("%d%d",&N,&M);
for(int i=;i<=N+;i++){
val[i]=i-;
}
val[]=-INF;
val[N+]=INF;
root=init(,N+,);
for(int i=;i<=M;i++){
int ii,jj;
scanf("%d%d",&ii,&jj);
int l=findx(ii),r=findx(jj+);
splay(l,);
splay(r,l);
tag[son[son[root][]][]]^=;
}
dfs(root);
return ;
}
splay区间翻转的更多相关文章
- hdu-1890-Robotic Sort splay区间翻转
题意: 依次找第i大的数下标pos[i],然后将区间[i,pos[i]]翻转 分析: splay树区间翻转 // File Name: ACM/HDU/1890.cpp // Author: Zlbi ...
- bzoj 3223 文艺平衡树 splay 区间翻转
Tyvj 1728 普通平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 17715 Solved: 7769[Submit][Status][ ...
- bzoj 1251序列终结者 splay 区间翻转,最值,区间更新
序列终结者 Time Limit: 20 Sec Memory Limit: 162 MBSubmit: 4594 Solved: 1939[Submit][Status][Discuss] De ...
- BZOJ 3223 Splay区间翻转
思路: 区间翻转的裸题 终于tm理解splay了-- //By SiriusRen #include <cstdio> #include <cstring> #include ...
- Splay(区间翻转)&树套树(Splay+线段树,90分)
study from: https://tiger0132.blog.luogu.org/slay-notes P3369 [模板]普通平衡树 #include <cstdio> #inc ...
- hdu1890 Robotic Sort (splay+区间翻转单点更新)
Problem Description Somewhere deep in the Czech Technical University buildings, there are laboratori ...
- 2019牛客国庆集训派对day1 K题 双向链表练习题 splay区间翻转
题目链接: 解法: 先建n颗平衡树,合并的时候将a中最右的结点翻转到根节点,b中最左的结点翻转到根节点,对合并后的根节点进行标记. #include <bits/stdc++.h> usi ...
- hdu 1890 Robotic Sort(splay 区间反转+删点)
题目链接:hdu 1890 Robotic Sort 题意: 给你n个数,每次找到第i小的数的位置,然后输出这个位置,然后将这个位置前面的数翻转一下,然后删除这个数,这样执行n次. 题解: 典型的sp ...
- bzoj3223 Tyvj 1729 文艺平衡树(Splay Tree+区间翻转)
3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 2202 Solved: 1226[Submit][Sta ...
随机推荐
- 面试系列32 集群部署时的分布式session如何实现
session是啥?浏览器有个cookie,在一段时间内这个cookie都存在,然后每次发请求过来都带上一个特殊的jsessionid cookie,就根据这个东西,在服务端可以维护一个对应的sess ...
- PHP算法之宝石与石头
给定字符串J 代表石头中宝石的类型,和字符串 S代表你拥有的石头. S 中每个字符代表了一种你拥有的石头的类型,你想知道你拥有的石头中有多少是宝石. J 中的字母不重复,J 和 S中的所有字符都是字母 ...
- static,final关键字,Object类的tostring方法,equals方法,hashCode方法
1)static关键字 static可以修饰:属性.方法.代码块 静态方法不能访问非静态 属性 或 方法 属性(变量): 成员变量: 静态变量: 通过 类名.静态变量来访问 通过 对象名.静态变量来访 ...
- 8.0后广播在AndroidManifest.xml中注册后发送intent接收不到广播
8.0后广播在AndroidManifest.xml中注册后发送intent是接收不到广播了,看了一下原因,好像是8.0为了管理系统和节约电量特别针对广播和服务发送intent的方式启动做出的改变,也 ...
- tomcat的webapps下面包含五个自带的项目
1.docs tomcat的介绍和操作文档等 2.examples 小程序示例 3.host-manager host管理 4.manager(重点) 进行 Server Status 和 Appli ...
- Codeforces Round #563 (Div. 2) F. Ehab and the Big Finale
后续: 点分治标程 使用father数组 比使用vis数组优秀(不需要对vis初始化) https://codeforces.com/problemset/problem/1174/F https:/ ...
- python3 enum模块
枚举是绑定到唯一的常量值的一组符号名称(成员).在枚举中,成员可以通过身份进行比较,枚举本身可以迭代. 1.Enum模块 该模块定义了四个枚举类,可用于定义唯一的名称和值集:Enum,IntEnum, ...
- day29 面向对象入门
Python之路,Day17 = Python基础17-面向对象入门 创建类和对象 面向对象编程是一种编程方式,此编程方式的落地需要使用 “类” 和 “对象” 来实现,所以,面向对象编程其实就是对 “ ...
- poj 1742 Coins(二进制优化多重背包)
传送门 解题思路 多重背包,二进制优化.就是把每个物品拆分成一堆连续的\(2\)的幂加起来的形式,然后把最后剩下的也当成一个元素.直接类似\(0/1\)背包的跑就行了,时间复杂度\(O(nmlogc) ...
- LUOGU P4042 [AHOI2014/JSOI2014]骑士游戏 (spfa+dp)
传送门 解题思路 首先设\(f[x]\)表示消灭\(x\)的最小花费,那么转移方程就是 \(f[x]=min(f[x],\sum f[son[x]] +s[x])\),如果这个转移是一个有向无环图,那 ...