P2596 [ZJOI2006]书架
思路
一开始写fhq-treap
感觉越写越感觉splay好些,就去splay
然后维护序列
注意前驱后继的不存在的情况
但不用插入虚拟节点(那插入岂不太麻烦)
跑的真慢的一批,splay太多了
错误
好多错误
只好对拍
代码
//这个题用treap似乎小题大做了,所以我用splay
#include <iostream>
#include <cstdio>
#define FOR(i,a,b) for(int i=a;i<=b;++i)
using namespace std;
const int maxn=2e5+7;
int read() {
int x=0,f=1;char s=getchar();
for(;s>'9'||s<'0';s=getchar()) if(s=='-') f=-1;
for(;s>='0'&&s<='9';s=getchar()) x=x*10+s-'0';
return x*f;
}
int n,m,a[maxn],rt,cnt,pos[maxn];
struct node {
int ch[2],fa,val,siz;
}e[maxn];
void pushup(int x) {
e[x].siz=e[e[x].ch[1]].siz+e[e[x].ch[0]].siz+1;
}
void rotate(int x) {
int y=e[x].fa,z=e[y].fa,k=(e[y].ch[1]==x);
e[z].ch[e[z].ch[1]==y]=x;
e[x].fa=z;
e[y].ch[k]=e[x].ch[k^1];
e[e[x].ch[k^1]].fa=y;
e[y].fa=x;
e[x].ch[k^1]=y;
pushup(x),pushup(y);
}
void splay(int x,int goal) {
while(e[x].fa!=goal) {
int y=e[x].fa,z=e[y].fa;
if(z!=goal) (e[y].ch[0]==x)^(e[z].ch[0]==y) ? rotate(x):rotate(y);
rotate(x);
}
if(goal==0) rt=x;
}
int k_th(int k) {
int now=rt;
while(233) {
if(e[e[now].ch[0]].siz+1==k) return e[now].val;
if(e[e[now].ch[0]].siz>=k) now=e[now].ch[0];
else k-=e[e[now].ch[0]].siz+1,now=e[now].ch[1];
}
splay(now,0);
}
int build(int l,int r,int f) {
if(l>r) return 0;
int mid=(l+r)>>1,p=(++cnt);
e[p].fa=f;
e[p].siz=1;
e[p].val=a[mid];
pos[a[mid]]=cnt;
e[p].ch[0]=build(l,mid-1,p);
e[p].ch[1]=build(mid+1,r,p);
pushup(p);
return p;
}
void dfs(int now) {
if(!now) return;
dfs(e[now].ch[0]);
cout<<e[now].val<<" ";
dfs(e[now].ch[1]);
}
int qq(int x) {
splay(x,0);
if(e[rt].val<e[x].val) return rt;
int now=e[rt].ch[0];
if(!now) return 0;
while(e[now].ch[1]) now=e[now].ch[1];
splay(now,0);
return now;
}
int hj(int x) {//x是pos
splay(x,0);
if(e[rt].val>e[x].val) return rt;
int now=e[rt].ch[1];
if(!now) return 0;
while(e[now].ch[0]) now=e[now].ch[0];
splay(now,0);
return now;
}
int p;
void delet(int x) {
int last=qq(x),nxt=hj(x);
if(last) splay(last,0);
if(nxt) splay(nxt,last);
if(nxt==0) {
p=e[last].ch[1];
e[last].ch[1]=0;
} else {
p=e[nxt].ch[0];
e[nxt].ch[0]=0;
if(nxt) splay(nxt,0);
}
}
void insert(int k) {
int now=rt;
while(e[now].ch[k]) now=e[now].ch[k];
e[now].ch[k]=p;
e[p].siz=1;
e[p].fa=now;
splay(p,0);
}
int main() {
n=read(),m=read();
FOR(i,1,n) a[i]=read();
rt=build(1,n,0);
char s[20];
FOR(i,1,m) {
cin>>s;
if(s[0]=='T') {
int a=read();
delet(pos[a]);
insert(0);
} else
if(s[0]=='B') {
int a=read();
delet(pos[a]);
insert(1);
} else
if(s[0]=='I') {
int a=read(),k=read(),b;
if(k==0) continue;
if(k==-1) b=qq(pos[a]);
else b=hj(pos[a]);
if(!b) continue;
int x=a,y=e[b].val;
swap(pos[x],pos[y]);
swap(e[pos[x]].val,e[pos[y]].val);
} else
if(s[0]=='A') {
int a=read();
splay(pos[a],0);
cout<<e[e[rt].ch[0]].siz<<"\n";
} else
if(s[0]=='Q') {
int a=read();
cout<<k_th(a)<<"\n";
}
}
return 0;
}
P2596 [ZJOI2006]书架的更多相关文章
- 洛谷 P2596 [ZJOI2006]书架 解题报告
P2596 [ZJOI2006]书架 题目描述 小T有一个很大的书柜.这个书柜的构造有些独特,即书柜里的书是从上至下堆放成一列.她用1到n的正整数给每本书都编了号. 小T在看书的时候,每次取出一本书, ...
- P2596 [ZJOI2006]书架 && Splay 区间操作(三)
P2596 [ZJOI2006]书架 题目描述 小T有一个很大的书柜.这个书柜的构造有些独特,即书柜里的书是从上至下堆放成一列.她用1到n的正整数给每本书都编了号. 小T在看书的时候,每次取出一本书, ...
- fhq_treap || BZOJ1861: [Zjoi2006]Book 书架 || Luogu P2596 [ZJOI2006]书架
题面:P2596 [ZJOI2006]书架 题解:记录每本书对应的节点编号 普通fhq_treap无法查询一个权值的排名,所以在普通fhq_treap上多记录每个节点的父亲(可加在pushup函数中) ...
- [洛谷P2596] [ZJOI2006]书架
洛谷题目链接:书架 题目描述 小T有一个很大的书柜.这个书柜的构造有些独特,即书柜里的书是从上至下堆放成一列.她用1到n的正整数给每本书都编了号. 小T在看书的时候,每次取出一本书,看完后放回书柜然后 ...
- P2596 [ZJOI2006]书架(splay)
[题目链接] https://www.luogu.org/problemnew/show/P2596 平衡树,需支持五个操作: 1. 将某元素置顶:将元素旋到根,然后将左子树合并到该元素的后继 2. ...
- luogu P2596 [ZJOI2006]书架
传送门 感觉要死在\(Splay\)里了 orz 这题用\(Splay\)维护这个序列,其中的第\(k\)大点代表这个序列的第\(k\)个数 第一个操作,先把那个数所在的点旋到根,然后把整个根的左子树 ...
- 洛谷 P2596 [ZJOI2006]书架 (splay)
题目描述 小T有一个很大的书柜.这个书柜的构造有些独特,即书柜里的书是从上至下堆放成一列.她用1到n的正整数给每本书都编了号. 小T在看书的时候,每次取出一本书,看完后放回书柜然后再拿下一本.由于这些 ...
- 「luogu2569」[ZJOI2006] 书架
「luogu2569」[ZJOI2006]书架 题目大意 给定一个长度为 \(n\) 序列,序列中第 \(i\) 个元素有编号 \(a_i(a_i \in \Z \cap [1,n])\),需要支持五 ...
- [Luogu 2596] ZJOI2006 书架
[Luogu 2596] ZJOI2006 书架 第一次指针写 FHQ_Treap(省选噩梦数据结构)AC 啦! 省选试机写它,紧张过度失败了. 省选 Day 1 考场写它,写挂了. 省选 Day 1 ...
随机推荐
- TP条件查询和分页查询
一.条件查询 前端页面 <!doctype html> <html> <head> <meta charset="utf-8"> & ...
- 《大话设计模式》c++实现 之策略模式
一.UML图 二.概念 策略模式:他定义了算法家族,分别封装起来,让他们之间可以互相替换,此模式让算法的变化,不会影响到使用算法的客户. 三.优点 (1)策略模式是一种定义一系列算法的方法,从 ...
- Python全栈-网络编程基础
一.C/S架构 1.硬件C/S架构 如PC-打印机 2.软件C/S架构 如PC-网站服务器 参照: https://baike.baidu.com/item/Client%2FServer/15044 ...
- Python全栈-day14-模块和包
一.模块 1.模块 1)定义 一系列功能的集合体,在Python中py文件就是一个模块 2)模块的类别 a.使用Python编写的py文件 b.已经被编译成共享库或者DLL的C 或者 C++ 扩展 c ...
- python二 总结--函数-- 装饰器
装饰器是什么? 有什么用? 为什么要用? 真的有用吗? 1.装饰器: 装饰器: 定义:本质是函数,(装饰其他函数)就是为其他函数添加附加功能. 原则:1.不能修改被装饰的函数的源代码 ...
- sqlmap的使用
安全测试===sqlmap(壹)转载 六.优化 这些参数可以优化Sqlmap的性能. 1.一键优化 参数:-o 添加此参数相当于同时添加下列三个优化参数: --keep-alive --null- ...
- NGINX的几个应用场景
NGINX的几个应用场景 两个参考地址: NGINX的百度百科:https://baike.baidu.com/item/nginx/3817705?fr=aladdin NGINX的中文网站:htt ...
- c# 定义委托和使用委托(事件的使用)
使用委托时要先实例化,和类一样,使用new关键字产生委托的新实例,然后将一个或者多个与委托签名匹配的方法与委托实例关联.随后调用委托时,就会调用所有与委托实例关联的方法. 与委托关联可以是任何类或者结 ...
- onclick 常用手册
1.如何去使用onclick来跳转到我们指定的页面/跳转到指定url ☆如果只是在本页显示的话,可以直接用location, 方法如下: ①onclick="javascript:windo ...
- 初学delphi
今天女朋友的一门课,要求用delphi 软件编程,内容是一个计算器.当然,这个工作肯定是落在我的头上了. 这个软件是我第一次使用,边自学边进行代码编写,在n多次修改完善之后,终于成形.功能不是很多,跟 ...