1, splay的一些基本操作.

  • 使用前要插入$-INF,+INF$保证每个点的前驱后继存在.
  • $get$函数在$x$存在时, 调用后, 根为$x$, 否则根为$x$的前驱或后继
const int N = 1e6+10;
int n, tot, rt, sz;
struct {
int cnt,sz,fa,ch[2],v;
} tr[N];
void pu(int x) {
tr[x].sz=tr[tr[x].ch[0]].sz+tr[tr[x].ch[1]].sz+tr[x].cnt;
}
void rot(int x) {
int y=tr[x].fa,z=tr[y].fa;
int f=tr[y].ch[1]==x;
tr[z].ch[tr[z].ch[1]==y]=x,tr[x].fa=z;
tr[y].ch[f]=tr[x].ch[f^1],tr[tr[x].ch[f^1]].fa=y;
tr[x].ch[f^1]=y,tr[y].fa=x,pu(y);
}
void splay(int x, int s=0) {
for (int y; y=tr[x].fa,y!=s; rot(x)) if (tr[y].fa!=s) {
rot((tr[y].ch[0]==x)==(tr[tr[y].fa].ch[0]==y)?y:x);
}
if (!s) rt=x;
}
void get(int x) {
int cur=rt;
while (x!=tr[cur].v&&tr[cur].ch[x>tr[cur].v]) cur=tr[cur].ch[x>tr[cur].v];
splay(cur);
}
void insert(int x) {
int cur=rt,p=0;
while (cur&&x!=tr[cur].v) p=cur,cur=tr[cur].ch[x>tr[cur].v];
if (cur) ++tr[cur].cnt;
else {
cur=++tot;
if (p) tr[p].ch[x>tr[p].v]=cur,tr[cur].fa=p;
tr[cur].v=x,tr[cur].sz=tr[cur].cnt=1;
}
splay(cur);
}
int pre(int x) {
get(x);
if (tr[rt].v<=x) return rt;
int cur=tr[rt].ch[0];
while (tr[cur].ch[1]) cur=tr[cur].ch[1];
return cur;
}
int nxt(int x) {
get(x);
if (tr[rt].v>=x) return rt;
int cur=tr[rt].ch[1];
while (tr[cur].ch[0]) cur=tr[cur].ch[0];
return cur;
}
void erase(int x) {
int s1=pre(x-1),s2=nxt(x+1);
splay(s1),splay(s2,s1);
int &cur=tr[s2].ch[0];
if (tr[cur].cnt>1) --tr[cur].cnt,splay(cur);
else cur=0;
}

2, splay插入区间,区间翻转等操作.

这时候splay维护的是每个下标对应的权值, 下标通过第k大来查询

  • 使用前要调用$build(a,0,rt,1,2);$
const int N = 1e6+10;
int n, rt, tot;
int a[N];
struct _ {
int sz,v,ch[2],fa,rev;
} tr[N];
void pu(int o) {
tr[o].sz=tr[tr[o].ch[0]].sz+tr[tr[o].ch[1]].sz+1;
}
void pd(int o) {
if (tr[o].rev) {
swap(tr[o].ch[0],tr[o].ch[1]);
tr[tr[o].ch[0]].rev^=1;
tr[tr[o].ch[1]].rev^=1;
tr[o].rev=0;
}
}
void rot(int x) {
int y=tr[x].fa,z=tr[y].fa;
int f=tr[y].ch[1]==x;
tr[z].ch[tr[z].ch[1]==y]=x,tr[x].fa=z;
tr[y].ch[f]=tr[x].ch[f^1],tr[tr[x].ch[f^1]].fa=y;
tr[x].ch[f^1]=y,tr[y].fa=x,pu(y);
}
void splay(int x, int s=0) {
for (int y; y=tr[x].fa,y!=s; rot(x)) if (tr[y].fa!=s) {
rot((tr[y].ch[0]==x)==(tr[tr[y].fa].ch[0]==y)?y:x);
}
if (!s) rt=x;
}
int find(int x, int k) {
pd(x); int s=tr[tr[x].ch[0]].sz;
if (k==s+1) return x;
if (k<=s) return find(tr[x].ch[0],k);
return find(tr[x].ch[1],k-s-1);
}
void build(int *a, int f, int &o, int l, int r) {
if (l>r) return;
o = ++tot;
tr[o].v = a[mid], tr[o].fa = f;
build(s,o,tr[o].ch[0],l,mid-1);
build(s,o,tr[o].ch[1],mid+1,r);
pu(o);
}
void ins(int x, int n) {
build(a,0,p,1,n);
int s1=find(rt,x-1), s2=find(rt,x);
splay(s1),splay(s2,s1);
tr[s2].ch[0]=p,tr[p].fa=s2;
pu(p),pu(s2);
}
void del(int x, int n) {
int s1=find(rt,x-1), s2=find(rt,x+n);
splay(s1),splay(s2,s1);
tr[s2].ch[0]=0;
pu(s1),pu(s2);
}
void reverse(int x, int n) {
int s1=find(rt,x-1), s2=find(rt,x+n);
splay(s1),splay(s2,s1);
tr[tr[s2].ch[0]].rev^=1;
}

splay板子的更多相关文章

  1. [bzoj] 1588 营业额统计 || Splay板子题

    原题 给出一个n个数的数列ai ,对于第i个元素ai定义\(fi=min(|ai-aj|) (1<=j<i)\),f1=a1,求\(/sumfi\) Splay板子题. Splay讲解:h ...

  2. POJ - 3481 splay板子

    Double Queue 默写splay板子 很多细节问题... #include<cstdio> #include<iostream> using namespace std ...

  3. 个人整理的数组splay板子,指针的写的太丑了就不放了。。

    splay的板子.. 由于被LCT榨干了..所以昨天去学了数组版的splay,现在整理一下板子.. 以BZOJ3224和3223为例题..暂时只有这些,序列的话等有时间把维修序列给弄上来!! BZOJ ...

  4. bzoj3224 splay板子

    开始学习新知识:splay——tree 是个板子题,学习splay可以看博客 https://blog.csdn.net/Clove_unique/article/details/50630280 # ...

  5. BZOJ 3224 Tyvj 1728 普通平衡树 | Splay 板子+SPlay详细讲解

    下面给出Splay的实现方法(复杂度证明什么的知道是 nlogn 就可以啦) 首先对于一颗可爱的二叉查找树,是不能保证最坏nlogn的复杂度(可以想象把一个升序序列插入) (二叉查找树保证左子树元素大 ...

  6. BZOJ[NOI2004]郁闷的出纳员 | Splay板子题

    题目: 洛谷也能评测....还有我wa了10多次的记录233 题解: 不要想得太复杂,搞一个全局变量记录一下工资的改变量Delta,这样可以等询问的时候就输出val+Delta,然后插入的时候插入x- ...

  7. P3369 【模板】普通平衡树(splay)

    P3369 [模板]普通平衡树 就是不用treap splay板子,好好背吧TAT #include<iostream> #include<cstdio> #include&l ...

  8. 【题解】 [HNOI2004]宠物收养场(Splay)

    懒得复制,戳我戳我 Solution: \(Splay\)板子,注意交换的地方,然后就是注意不要越界node[x],应该是\(node[now]\),其次就是数组可以开大点 Code: //It is ...

  9. 【题解】 [HNOI2002]营业额统计 (Splay)

    懒得复制,戳我戳我 Solution: \(Splay\)板子题,注意可以选择相等大小 Code: //It is coded by Ning_Mew on 4.10 #include<bits ...

随机推荐

  1. python学习读取配置文件

    配置文件作为一种可读性很好的格式,非常适用于存储程序中的配置数据. 在每个配置文件中,配置数据会被分组(比如“config”和 “cmd”). 每个分组在其中指定对应的各个变量值.如下: # 定义co ...

  2. USB开发库STSW-STM32121文件分析

    hw_config.c: 该文件中包含系统配置的函数. usb_desc.c:各种描述符 usb-endp.c:就两个函数分别处理端点1的IN和端点2的OUT. usb_istr.c: 该文件中只有一 ...

  3. 根据wsdl文件,Web工程自动生成webservice客户端调用

    根据wsdl文件,Web工程自动生成webservice客户端调用 1,工具:带有webservice插件的eclips 2,步骤: (1),新建一个Web工程:WSDLTest (2),浏览器访问W ...

  4. djando 项目用django自己服务器在局域网中被访问设置

    这是一个相当操蛋的东西,害老子搞了那么久,其实嘞,也用不着那么恨,都是自己做的孽!! -----------------人工分割线----------------------------------- ...

  5. Linux 进程学习笔记

    1.什么是程序?什么是进程?它们有什么区别? 定义: 程序:程序(Program)是一个静态的命令集合,程序一般放在磁盘中,然后通过用户的执行来触发.触发后程序会加载到内存中成为一个个体,就是进程. ...

  6. 20145302张薇 《网络对抗技术》 web基础

    20145302张薇 <网络对抗> web基础 实验问题回答 1.什么是表单 表单在网页中主要负责数据采集功能:一般网页上需要用户输入.选择的地方都会用到表单 表单标签:即,用于确定表单所 ...

  7. 前向算法Python实现

    前言 这里的前向算法与神经网络里的前向传播算法没有任何联系...这里的前向算法是自然语言处理领域隐马尔可夫模型第一个基本问题的算法. 前向算法是什么? 这里用一个海藻的例子来描述前向算法是什么.网上有 ...

  8. 通过jquery获得某个元素的位置, 透明div, 弹出框, 然后在旁边显示toggle子级联菜单-hover的bug解决

    jquery的"筛选选择器", 都是用冒号开头的, 即, 冒号选择器就是 筛选选择器.如: :first, :last, :eq(index), :first-child,...等 ...

  9. BZOJ 1044: [HAOI2008]木棍分割 DP 前缀和优化

    题目链接 咳咳咳,第一次没大看题解做DP 以前的我应该是这样的 哇咔咔,这tm咋做,不管了,先看个题解,再写代码 终于看懂了,卧槽咋写啊,算了还是抄吧 第一问类似于noip的那个跳房子,随便做 这里重 ...

  10. HDU 1811(并查集+拓扑排序)题解

    Problem Description 自从Lele开发了Rating系统,他的Tetris事业更是如虎添翼,不久他遍把这个游戏推向了全球.为了更好的符合那些爱好者的喜好,Lele又想了一个新点子:他 ...