题目类型:平衡树

传送门:>Here<

题意:要求维护一个数列,支持:将某个元素置顶或置底,交换某元素与其前驱或后继的位置,查询编号为\(S\)的元素的排名,查询排名第\(k\)的元素编号

解题思路

可以说是平衡树维护数列的入门题。当平衡树在维护数列时,关键字是在数列中的排名。因此中序遍历即为当前数列。注意在平衡树维护数列中,会涉及到两个编号。一个编号是这个节点在平衡树内的编号,一般外界是不会直接访问的。另一个是题目赋予的编号,代表这个位置数列上对应的值。外部编号即为一个附加值。当我们需要找到外部编号为\(S\)的元素,好像比较麻烦。此时,我们需要多开一个数组\(pos\),令\(pos[idx[o]]=o\),相当于是\(idx\)数组的逆运算。在知道一个外部编号时,能够迅速找到其所对应的节点编号。然后就可以对其进行直接操作了

至于前驱后继交换,我们只需要交换这两个节点所对应的外部编号以及更新\(pos\),没有必要交换平衡树内的节点。

Code

/*By DennyQi 2018*/
#include <cstdio>
#include <queue>
#include <cstring>
#include <algorithm>
#define r read()
using namespace std;
typedef long long ll;
const int MAXN = 80010;
const int INF = 1061109567;
inline int Max(const int a, const int b){ return (a > b) ? a : b; }
inline int Min(const int a, const int b){ return (a < b) ? a : b; }
inline int read(){
int x = 0; int w = 1; register char c = getchar();
for(; c ^ '-' && (c < '0' || c > '9'); c = getchar());
if(c == '-') w = -1, c = getchar();
for(; c >= '0' && c <= '9'; c = getchar()) x = (x<<3) + (x<<1) + c - '0'; return x * w;
}
int N,M,S,T;
char opt[10];
int idx[MAXN],pos[MAXN];
struct Splay{
int fa[MAXN],ch[MAXN][2],size[MAXN],num_node,rt;
inline void update(int x){
size[x] = 1;
if(ch[x][0]) size[x] += size[ch[x][0]];
if(ch[x][1]) size[x] += size[ch[x][1]];
}
inline bool rson(int f, int x){
return ch[f][1] == x;
}
inline void rotate(int x){
int f = fa[x], gf = fa[f];
bool p = rson(f, x), q = !p;
if(!gf) rt = x; else ch[gf][rson(gf,f)] = x;
fa[x] = gf;
ch[f][p] = ch[x][q], fa[ch[x][q]] = f;
ch[x][q] = f, fa[f] = x;
update(f), update(x);
}
inline void splay(int x, int target){
int f,gf;
while(fa[x] != target){
f = fa[x], gf = fa[f];
if(gf == target){
rotate(x);
break;
}
if(rson(gf,f) ^ rson(f,x)) rotate(x); else rotate(f);
rotate(x);
}
}
inline void init(int id){
if(!rt){
rt = ++num_node;
idx[num_node] = id;
pos[id] = num_node;
size[num_node] = 1;
return;
}
int pre_rt = rt;
rt = ++num_node;
ch[rt][0] = pre_rt;
fa[pre_rt] = rt;
size[rt] = size[pre_rt] + 1;
idx[rt] = id;
pos[id] = rt;
}
inline void top(int s){
int o = pos[s];
splay(o, 0);
if(!ch[rt][0] && !ch[rt][1]) return;
int p = ch[rt][1];
if(!p){
ch[rt][1] = ch[rt][0];
ch[rt][0] = 0;
splay(ch[rt][1], 0);
return;
}
while(ch[p][0]) p = ch[p][0];
splay(p, rt);
ch[p][0] = ch[rt][0];
fa[ch[rt][0]] = p;
ch[rt][0] = 0;
update(p), update(rt);
}
inline void bottom(int s){
int o = pos[s];
splay(o, 0);
if(!ch[rt][0] && !ch[rt][1]) return;
int p = ch[rt][0];
if(!p){
ch[rt][0] = ch[rt][1];
ch[rt][1] = 0;
splay(ch[rt][0], 0);
return;
}
while(ch[p][1]) p = ch[p][1];
splay(p, rt);
ch[p][1] = ch[rt][1];
fa[ch[rt][1]] = p;
ch[rt][1] = 0;
update(p), update(rt);
}
inline int ask(int s){
int o = pos[s];
splay(o, 0);
return size[ch[rt][0]];
}
inline int query(int k){
int o = rt;
while(o){
if(size[ch[o][0]] >= k){
o = ch[o][0];
}
else if(size[ch[o][0]] + 1 < k){
k -= size[ch[o][0]] + 1;
o = ch[o][1];
}
else{
return idx[o];
}
}
return -1;
}
inline void insert(int S, int T){
if(T == 0) return;
splay(pos[S], 0);
if(T == -1){
int o = ch[rt][0];
while(ch[o][1]) o = ch[o][1];
splay(o, rt);
swap(idx[rt], idx[o]);
pos[idx[rt]] = rt, pos[idx[o]] = o;
update(o), update(rt);
}
if(T == 1){
int o = ch[rt][1];
while(ch[o][0]) o = ch[o][0];
splay(o, rt);
swap(idx[rt], idx[o]);
pos[idx[rt]] = rt, pos[idx[o]] = o;
update(o), update(rt);
}
}
}qxz;
int main(){
scanf("%d %d", &N, &M);
for(int i = 1; i <= N; ++i){
scanf("%d", &S);
qxz.init(S);
}
while(M--){
scanf("%s %d", opt, &S);
if(opt[0] == 'T'){
qxz.top(S);
}
if(opt[0] == 'B'){
qxz.bottom(S);
}
if(opt[0] == 'A'){
printf("%d\n", qxz.ask(S));
}
if(opt[0] == 'Q'){
printf("%d\n", qxz.query(S));
}
if(opt[0] == 'I'){
scanf("%d", &T);
qxz.insert(S,T);
}
}
}

☆ [ZJOI2006] 书架 「平衡树维护数列」的更多相关文章

  1. ☆ [HNOI2012] 永无乡 「平衡树启发式合并」

    题目类型:平衡树启发式合并 传送门:>Here< 题意:节点可以连边(不能断边),询问任意两个节点的连通性与一个连通块中排名第\(k\)的节点 解题思路 如果不需要询问排名,那么并查集即可 ...

  2. 「NOI2005」维护数列

    「NOI2005」维护数列 传送门 维护过程有点像线段树. 但我们知道线段树的节点并不是实际节点,而平衡树的节点是实际节点. 所以在向上合并信息时要加入根节点信息. 然后节点再删除后编号要回退(栈), ...

  3. 「luogu2569」[ZJOI2006] 书架

    「luogu2569」[ZJOI2006]书架 题目大意 给定一个长度为 \(n\) 序列,序列中第 \(i\) 个元素有编号 \(a_i(a_i \in \Z \cap [1,n])\),需要支持五 ...

  4. Luogu P2042 [NOI2005]维护数列(平衡树)

    P2042 [NOI2005]维护数列 题意 题目描述 请写一个程序,要求维护一个数列,支持以下\(6\)种操作:(请注意,格式栏中的下划线'_'表示实际输入文件中的空格) 输入输出格式 输入格式: ...

  5. 数据结构(Splay平衡树):COGS 339. [NOI2005] 维护数列

    339. [NOI2005] 维护数列 时间限制:3 s   内存限制:256 MB [问题描述] 请写一个程序,要求维护一个数列,支持以下 6 种操作:(请注意,格式栏 中的下划线‘ _ ’表示实际 ...

  6. 洛谷 P2596 [ZJOI2006]书架 解题报告

    P2596 [ZJOI2006]书架 题目描述 小T有一个很大的书柜.这个书柜的构造有些独特,即书柜里的书是从上至下堆放成一列.她用1到n的正整数给每本书都编了号. 小T在看书的时候,每次取出一本书, ...

  7. 企业运营对 DevOps 的「傲慢与偏见」

    摘要:出于各种原因,并非所有人都信任 DevOps .有些人觉得 DevOps 只不过给开发者改善产品提供了一个途径而已,还有的人觉得 DevOps 是一堆悦耳的空头支票,甚至有人认为 DevOps ...

  8. 「造个轮子」——cicada 设计一个配置模块

    前言 在前两次的 cicada 版本中其实还不支持读取配置文件,比如对端口.路由的配置. 因此我按照自己的想法创建了一个 issue ,也收集到了一些很不错的建议. 最终其实还是按照我之前的想法来做了 ...

  9. 「造个轮子」——cicada 源码分析

    前言 两天前写了文章<「造个轮子」--cicada(轻量级 WEB 框架)> 向大家介绍了 cicada 之后收到很多反馈,也有许多不错的建议. 同时在 GitHub 也收获了 80 几颗 ...

随机推荐

  1. 自然语言处理(nlp)比计算机视觉(cv)发展缓慢,而且更难!

    https://mp.weixin.qq.com/s/kWw0xce4kdCx62AflY6AzQ 1.  抢跑的nlp nlp发展的历史非常早,因为人从计算机发明开始,就有对语言处理的需求.各种字符 ...

  2. java线程介绍

    文章讲解要点 1.线程创建几种方式2.线程常见设置方法,包括优先级.优先级休眠.停止等3.多线程间的数据交互与锁机制4.项目源码下载   线程介绍.png 一.线程创建方式 常见的线程创建方法以下三种 ...

  3. SQL Server 数据库调整表中列的顺序操作

    SQL Server 数据库中表一旦创建,我们不建议擅自调整列的顺序,特别是对应的应用系统已经上线,因为部分开发人员,不一定在代码中指明了列名.表是否可以调整列的顺序,其实可以自主设置,我们建议在安装 ...

  4. chart 模板 - 每天5分钟玩转 Docker 容器技术(165)

    Helm 通过模板创建 Kubernetes 能够理解的 YAML 格式的资源配置文件,我们将通过例子来学习如何使用模板. 以 templates/secrets.yaml 为例: 从结构看,文件的内 ...

  5. 一文把samba相关的都说清楚

    1.前言 samba源码都一样,配置也也一样,各个不同linux版本,唯一不同的是对服务的启动方式不同.下面以ubuntu14.4为例,说明. 2. 安装samba samba的安装,可以源码安装,大 ...

  6. Python函数(一)之杵臼之交

    Python函数 函数的作用:对功能进行封装,减少重复代码,方便维护,流程清晰明了,易于理解. 函数的结构: def 函数名():      函数体       return语句 函数的返回值: 可以 ...

  7. netcat的简单使用(一)

    简单写一下netcat这个强悍的工具,主要是怕自己忘了 功能大致这些个,有遗漏的欢迎私信补充 1.侦听模式/传输模式 2.telnet/获取banner信息 3.传输文本信息 4.传输文件/目录 5. ...

  8. 自动化测试之路2---python安装

    借鉴这位老哥的文章http://www.cnblogs.com/shabbylee/p/6792555.html

  9. SQLServer之创建DML AFTER INSERT触发器

    DML AFTER INSERT触发器创建原理 触发器触发时,系统自动在内存中创建deleted表或inserted表,内存中创建的表只读,不允许修改,触发器执行完成后,自动删除. insert触发器 ...

  10. (转载)最完整的自动化测试流程:Python编写执行测试用例及定时自动发送最新测试报告邮件

    今天笔者就要归纳总结下一整套测试流程,从无到有,实现零突破,包括如何编写测试用例,定时执行测试用例,查找最新生成的测试报告文件,自动发送最新测试报告邮件,一整套完整的测试流程.以后各位只要着重如何编写 ...