「luogu2569」[ZJOI2006]书架

题目大意

给定一个长度为 \(n\) 序列,序列中第 \(i\) 个元素有编号 \(a_i(a_i \in \Z \cap [1,n])\),需要支持五种操作:

  1. \(Top\) \(S\) ——表示把编号为 \(S\) 的书放在最上面;
  2. \(Bottom\) \(S\)——表示把编号为 \(S\) 的书放在最下面;
  3. \(Insert\) \(S\) \(T\)——\(T \in \{-1,0,1\}\),若编号为 \(S\) 的书上面有 \(X\) 本书,则这条命令表示把这本书放回去后它的上面有 \(X+T\) 本书;
  4. \(Ask\) \(S\)——询问编号为 \(S\) 的书的上面目前有多少本书;
  5. \(Query\) \(S\)——询问从上面数起的第 \(S\) 本书的编号。

对于每个 \(Ask\),\(Query\) 操作,输出答案。

(以上摘自luogu)

题解

\(fhq\_treap\) 复健,没有题解。

代码如下:

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cstdlib>
inline int in() {
int x=0;char c=getchar();bool f=false;
while(c<'0'||c>'9') f|=c=='-', c=getchar();
while(c>='0'&&c<='9') x=(x<<1)+(x<<3)+(c^48), c=getchar();
return f?-x:x;
} const int N = 8e4+5; int mp[N]; struct fhq_treap {
#define t a[p]
#define lson a[a[p].l]
#define rson a[a[p].r]
struct node {
int l, r, key, size, fa, rnd;
}a[N];
int tot, rt; inline int new_node(int key) {
a[++tot]=(node){0, 0, key, 1, 0, rand()};
mp[key]=tot;
return tot;
}
inline void push_up(int p) {
if(p) {
t.size=lson.size+rson.size+1;
lson.fa=rson.fa=p;
}
}
void split(int p, int &x, int &y, int k) {
if(!p) return (void)(x=y=0);
if(lson.size<k) x=p, split(t.r, t.r, y, k-lson.size-1);
else y=p, split(t.l, x, t.l, k);
push_up(p);
}
int merge(int x, int y) {
if(!x||!y) return x?x:y;
int p=0;
if(a[x].rnd<a[y].rnd) p=x, t.r=merge(t.r, y);
else p=y, t.l=merge(x, t.l);
push_up(p);
return p;
}
inline int kth_id(int k) {
int p=rt;
while(lson.size+1!=k)
if(lson.size>=k) p=t.l;
else k-=lson.size+1, p=t.r;
return a[p].key;
}
inline int find_rank(int p) {
int ret=lson.size+1;
for(;p!=rt;p=t.fa)
if(a[t.fa].r==p) ret+=a[a[t.fa].l].size+1;
return ret;
}
void debug(int p) {
if(t.l) debug(t.l);
printf(" %d", a[p].key);
if(t.r) debug(t.r);
}
#undef t
#undef lson
#undef rson
}T; int main() {
srand(20021111);
char opt[10];
int n=in(), m=in();
for(int i=1;i<=n;++i) T.rt=T.merge(T.rt, T.new_node(in())); int a, b, c, d, s, t, k;
while(m--) {
scanf("%s", opt), s=in();
if(opt[0]=='T') {
k=T.find_rank(mp[s]);
T.split(T.rt, a, b, k);
T.split(a, a, c, k-1);
T.rt=T.merge(T.merge(c, a), b);
}
else if(opt[0]=='B') {
k=T.find_rank(mp[s]);
T.split(T.rt, a, b, k);
T.split(a, a, c, k-1);
T.rt=T.merge(a, T.merge(b, c));
}
else if(opt[0]=='I') {
t=in();
k=T.find_rank(mp[s]);
if(t>0) {
T.split(T.rt, a, b, k);
T.split(a, a, c, k-1);
T.split(b, b, d, k-T.a[a].size-T.a[c].size+1);
T.rt=T.merge(T.merge(a, b), T.merge(c, d));
}
else if(t<0) {
T.split(T.rt, a, b, k-1);
T.split(a, a, c, k-2);
T.split(b, b, d, k-T.a[a].size-T.a[c].size);
T.rt=T.merge(T.merge(a, b), T.merge(c, d));
}
}
else if(opt[0]=='A') printf("%d\n", T.find_rank(mp[s])-1);
else printf("%d\n", T.kth_id(s));
//T.debug(T.rt);
//putchar('\n');
}
return 0;
}

「luogu2569」[ZJOI2006] 书架的更多相关文章

  1. 嘴巴题5 「BZOJ1864」[ZJOI2006] 三色二叉树

    1864: [Zjoi2006]三色二叉树 Time Limit: 1 Sec Memory Limit: 64 MB Submit: 1195 Solved: 882 [Submit][Status ...

  2. 「bzoj1003」「ZJOI2006」物流运输 最短路+区间dp

    「bzoj1003」「ZJOI2006」物流运输---------------------------------------------------------------------------- ...

  3. 「MoreThanJava」Java发展史及起航新世界

    「MoreThanJava」 宣扬的是 「学习,不止 CODE」,本系列 Java 基础教程是自己在结合各方面的知识之后,对 Java 基础的一个总回顾,旨在 「帮助新朋友快速高质量的学习」. 当然 ...

  4. 「译」JUnit 5 系列:条件测试

    原文地址:http://blog.codefx.org/libraries/junit-5-conditions/ 原文日期:08, May, 2016 译文首发:Linesh 的博客:「译」JUni ...

  5. 「译」JUnit 5 系列:扩展模型(Extension Model)

    原文地址:http://blog.codefx.org/design/architecture/junit-5-extension-model/ 原文日期:11, Apr, 2016 译文首发:Lin ...

  6. JavaScript OOP 之「创建对象」

    工厂模式 工厂模式是软件工程领域一种广为人知的设计模式,这种模式抽象了创建具体对象的过程.工厂模式虽然解决了创建多个相似对象的问题,但却没有解决对象识别的问题. function createPers ...

  7. 「C++」理解智能指针

    维基百科上面对于「智能指针」是这样描述的: 智能指针(英语:Smart pointer)是一种抽象的数据类型.在程序设计中,它通常是经由类型模板(class template)来实做,借由模板(tem ...

  8. 「JavaScript」四种跨域方式详解

    超详细并且带 Demo 的 JavaScript 跨域指南来了! 本文基于你了解 JavaScript 的同源策略,并且了解使用跨域跨域的理由. 1. JSONP 首先要介绍的跨域方法必然是 JSON ...

  9. 「2014-5-31」Z-Stack - Modification of Zigbee Device Object for better network access management

    写一份赏心悦目的工程文档,是很困难的事情.若想写得完善,不仅得用对工具(use the right tools),注重文笔,还得投入大把时间,真心是一件难度颇高的事情.但,若是真写好了,也是善莫大焉: ...

随机推荐

  1. scala的多种集合的使用(2)之集合常用方法

    一.常用的集合方法 1.可遍历集合的常用方法 下表列出了Traverable在所有集合常用的方法.接下来的符号: c代表一个集合 f代表一个函数 p代表一个谓词 n代表一个数字 op代表一个简单的操作 ...

  2. 如何把Office365的更新从半年通道改成月度通道

    转自msdn,转发链接:www.cnblogs.com/Charltsing/p/Office365month.html 作者QQ: 564955427 建立一个Bat文件,写入 下面内容 setlo ...

  3. Django ORM多表操作

    多表操作 创建模型 实例:我们来假定下面这些概念,字段和关系 作者模型:一个作者有姓名和年龄. 作者详细模型:把作者的详情放到详情表,包含生日,手机号,家庭住址等信息.作者详情模型和作者模型之间是一对 ...

  4. HashMap源码解读(jdk1.8)

    1.相关常量 默认初始化容量(大小) static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; 最大容量 static final int M ...

  5. Activiti6-FormService(学习笔记)重要

    设置流程定义文件: <?xml version="1.0" encoding="UTF-8"?> <definitions xmlns=&qu ...

  6. RedisCache 缓存

    /// <summary> /// 这是包装过公用的,用于本站而已. /// </summary> /// <author>rixiao</author> ...

  7. [洛谷P1006] 传纸条

    双线程DP的常识:DP的阶段问题:多阶段决策问题 传送门:$>here<$ 题意 n*m的矩阵里,从左上角走到右下角(只能往右或往下),再从右下角走回左上角(只能往左或往上).其中不能重复 ...

  8. 【dp】合唱队形

    题目描述 NN位同学站成一排,音乐老师要请其中的(N-KN−K)位同学出列,使得剩下的KK位同学排成合唱队形. 合唱队形是指这样的一种队形:设K位同学从左到右依次编号为1,2,…,K,他们的身高分别为 ...

  9. Floyd-蒟蒻也能看懂的弗洛伊德算法(当然我是蒟蒻)

    今天来讲点图论的知识,来看看最短路径的一个求法(所有的求法我以后会写,也有可能咕咕咕) 你们都说图看着没意思不好看,那今天就来点情景             暑假,_GC准备去一些城市旅游.有些城市之 ...

  10. xml转换为json格式时,如何将指定节点转换成数组 Json.NET

    使用Json.NET转换xml成json时,如果xml只有单个节点,但json要求是数组形式[], JsonConvert.SerializeXmlNode 并不能自动识别 示例如下: RecordA ...