思路

一开始写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]书架的更多相关文章

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

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

  2. P2596 [ZJOI2006]书架 && Splay 区间操作(三)

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

  3. fhq_treap || BZOJ1861: [Zjoi2006]Book 书架 || Luogu P2596 [ZJOI2006]书架

    题面:P2596 [ZJOI2006]书架 题解:记录每本书对应的节点编号 普通fhq_treap无法查询一个权值的排名,所以在普通fhq_treap上多记录每个节点的父亲(可加在pushup函数中) ...

  4. [洛谷P2596] [ZJOI2006]书架

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

  5. P2596 [ZJOI2006]书架(splay)

    [题目链接] https://www.luogu.org/problemnew/show/P2596 平衡树,需支持五个操作: 1. 将某元素置顶:将元素旋到根,然后将左子树合并到该元素的后继 2. ...

  6. luogu P2596 [ZJOI2006]书架

    传送门 感觉要死在\(Splay\)里了 orz 这题用\(Splay\)维护这个序列,其中的第\(k\)大点代表这个序列的第\(k\)个数 第一个操作,先把那个数所在的点旋到根,然后把整个根的左子树 ...

  7. 洛谷 P2596 [ZJOI2006]书架 (splay)

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

  8. 「luogu2569」[ZJOI2006] 书架

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

  9. [Luogu 2596] ZJOI2006 书架

    [Luogu 2596] ZJOI2006 书架 第一次指针写 FHQ_Treap(省选噩梦数据结构)AC 啦! 省选试机写它,紧张过度失败了. 省选 Day 1 考场写它,写挂了. 省选 Day 1 ...

随机推荐

  1. why big data

    很多人都知道大数据很火,就业很好,薪资很高,想往大数据方向发展.但该学哪些技术,学习路线是什么样的呢?用不用参加大数据培训呢?如果自己很迷茫,为了这些原因想往大数据方向发展,也可以,那么大讲台老师就想 ...

  2. caffe中通过prototxt文件查看神经网络模型结构的方法

    在修改propotxt之前我们可以对之前的网络结构进行一个直观的认识: 可以使用http://ethereon.github.io/netscope/#/editor 这个网址. 将propotxt文 ...

  3. css中块级元素、内联元素(行内元素、内嵌元素)

    Block element 块级元素    顾名思义就是以块显示的元素,高度宽度都是可以设置的.比如我们常用 的<div>.<p>.<ul>默认状态下都是属于块级元 ...

  4. 大数据处理框架之Strom: Storm----helloword

    大数据处理框架之Strom: Storm----helloword Storm按照设计好的拓扑流程运转,所以写代码之前要先设计好拓扑图.这里写一个简单的拓扑: 第一步:创建一个拓扑类含有main方法的 ...

  5. python单下划线与双下划线的区别

    Python 用下划线作为变量前缀和后缀指定特殊变量. _xxx 不能用'from moduleimport *'导入 __xxx__ 系统定义名字 __xxx 类中的私有变量名 核心风格:避免用下划 ...

  6. c# 类的知识

    在英语中类(class)是分类(classification)的根词.设计类的过程就是对信息进行分类,将相关信息放到有意义的实体中. 封装的目的: 封装就是定义类的重要原则.中心思想是:使用类的程序不 ...

  7. react+redux+react-router+node.js 开发实时聊天App 学习记录

    一.课程导学 1.React 主要解决的是UI层的问题,应用的状态需要借助Redux等状态管理. 2.前端React  +  antd-mobile UI组件库 + Redux 状态管理库 + Rea ...

  8. 转:【专题六】UDP编程

    引用: 前一个专题简单介绍了TCP编程的一些知识,UDP与TCP地位相当的另一个传输层协议,它也是当下流行的很多主流网络应用(例如QQ.MSN和Skype等一些即时通信软件传输层都是应用UDP协议的) ...

  9. SpringBoot整合redis哨兵主从服务

    前提环境: 主从配置 http://www.cnblogs.com/zwcry/p/9046207.html 哨兵配置 https://www.cnblogs.com/zwcry/p/9134721. ...

  10. HDU 1527 取石子游戏 (威佐夫博弈)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1527 有两堆石子,数量任意,可以不同.游戏开始由两个人轮流取石子.游戏规定,每次有两种不同的取法,一是 ...