题意:让你维护一个序列,支持以下6种操作:

  ADD x y d: 第x个数到第y个数加d 。

  REVERSE x y : 将区间[x,y]中的数翻转 。

  REVOLVE x y t :将区间[x,y]循环移位t次,如1 2 3 4 5 旋转2次后就变成4 5 1 2 3 。

  INSERT x p :在第x个数后面插入p 。

  DELETE x :删除第x个数 。

  MIN x y : 查询区间[x,y]中的最小值 。
思路:此题有反转区间和循环移位的操作,所以我们很容易可以想到用 splay FHQ_treap 来解决这个问题。splay的操作比较麻烦,细节较多,而FHQ_Treap 简单暴力,只比splay慢一点点。

FHQ_Treap可以做splay能做的所有操作(插入,删除一段序列,区间操作,维护区间的值等),并且这些操作只和merge(合并)和split(分裂)这2个函数有关,不需要考虑各种细节,比较无脑。

代码:

#include <cstdio>
#include <algorithm>
#include <iostream>
#include <cstring>
#include <ctime>
#define INF ~0U>>2
using namespace std;
const int maxn=500010;
//struct FHQ_Treap {
struct node {
node* ch[2];
int val, key, sz, mi, mark;
bool flip; node() {
val = INF;
mark = sz = 0;
mi = INF;
flip = 0;
key = rand();
} inline void update() {
sz = ch[0] -> sz + ch[1] ->sz + 1;
mi = min(ch[0] -> mi, min(ch[1] -> mi, val));
}
};
node *null = new node(), * root = null, *stack[maxn], *x, *last;
typedef pair<node*, node*> pnn;
int a[maxn], n; inline void maintain_flip(node* o) {
if (o == null) return;
o -> flip ^= 1;
} inline void maintain_mark(node* o, int c) {
if (o == null) return;
o -> val += c;
o -> mi += c;
o -> mark +=c;
} inline void pushdown(node* o) {
if (o == null) return;
if (o -> flip) {
o -> flip ^= 1;
maintain_flip(o -> ch[0]);
maintain_flip(o -> ch[1]);
swap(o -> ch[0], o -> ch[1]);
}
if (o -> mark) {
maintain_mark(o -> ch[0], o -> mark);
maintain_mark(o -> ch[1], o -> mark);
o -> mark = 0;
}
} inline node* newnode(int val) {
node *o = new node();
o ->ch[1] = o -> ch[0] = null;
o -> key = rand();
o -> val = o -> mi = val;
o -> sz = 1;
o -> flip = 0;
o -> mark = 0;
return o;
} node* merge(node* a, node* b) {
if (a == null) return b;
if (b == null) return a;
pushdown(a), pushdown(b);
if (a -> key < b -> key) {
a -> ch[1] = merge(a -> ch[1], b);
a ->update();
return a;
} else {
b -> ch[0] = merge(a, b -> ch[0]);
b -> update();
return b;
}
} pnn split(node* o, int rnk) {
if (o == null) return pnn(null, null);
pnn y;
pushdown(o);
if (o -> ch[0] -> sz >= rnk) {
y = split(o -> ch[0], rnk);
o -> ch[0] = y.second;
o -> update();
y.second = o;
}else {
y = split(o -> ch[1], rnk - o -> ch[0] -> sz -1);
o -> ch[1] = y.first;
o -> update();
y.first = o;
}
return y;
} inline node* build() {
int p = 0;
for(int i = 1;i <= n; i++) {
scanf("%d", &a[i]);
x = newnode(a[i]);
last = null;
while(p && stack[p] -> key > x -> key) {
stack[p] -> update();
last = stack[p];
stack[p--] = null;
}
if(p) stack[p] -> ch[1] = x;
x -> ch[0] = last;
stack[++p] = x;
}
while(p) stack[p--] -> update();
return stack[1];
} void del(node* o) {
if(o == null) return;
if(o -> ch[0] != null) del(o -> ch[0]);
if(o -> ch[1] != null) del(o -> ch[1]);
delete o;
} inline void insert() {
int pos, val;
scanf("%d%d",&pos,&val);
node* o = newnode(val);
pnn x = split(root, pos);
root = merge(merge(x.first, o), x.second);
} inline void erase() {
int pos;
scanf("%d", &pos);
pnn x = split(root, pos - 1);
pnn y = split(x.second, 1);
del(y.first);
root = merge(x.first, y.second);
} inline void reverse(void) {
int L, R;
scanf("%d%d",&L,&R);
pnn x = split(root, L - 1);
pnn y = split(x.second, R - L + 1);
maintain_flip(y.first);
root = merge(merge(x.first, y.first), y.second);
} inline void make_add(void) {
int L, R, c;
scanf("%d%d%d", &L, &R, &c);
pnn x = split(root, L - 1);
pnn y = split(x.second, R - L + 1);
maintain_mark(y.first, c);
root = merge(merge(x.first, y.first),y.second);
} inline int get_mi(void) {
int L, R;
scanf("%d%d",&L,&R);
if(n == 0) return 0;
pnn x = split(root, L - 1);
pnn y = split(x.second, R - L + 1);
pushdown(y.first);
int ans = y.first -> mi;
root = merge(merge(x.first, y.first),y.second);
return ans;
} inline void revove(void) {
int L, R, t;
scanf("%d%d%d", &L, &R, &t);
int k = R - L + 1;
t = (t % k + k) % k;
if(t == 0) return;
pnn x = split(root, L - 1);
pnn y = split(x.second, k);
pnn z = split(y.first, k - t);
root = merge(merge(merge(x.first, z.second),z.first),y.second);
}
//};
int main() {
int m;
scanf("%d", &n);
root = build();
char op[20];
scanf("%d", &m);
while(m--) {
scanf("%s", op + 1);
if (op[1] == 'I') {
insert();
} else if(op[1] == 'D') {
erase();
} else if(op[1] == 'A') {
make_add();
} else if(op[1] == 'M') {
printf("%d\n", get_mi());
} else if(op[1] == 'R') {
if(op[4] == 'E')
reverse();
else
revove();
}
}
}

  代码实现参考了这篇博客:http://www.cnblogs.com/LadyLex/p/7182631.html

POJ 3580 SuperMemo (FHQ_Treap)的更多相关文章

  1. poj 3580 SuperMemo

    题目连接 http://poj.org/problem?id=3580 SuperMemo Description Your friend, Jackson is invited to a TV sh ...

  2. POJ 3580 - SuperMemo - [伸展树splay]

    题目链接:http://poj.org/problem?id=3580 Your friend, Jackson is invited to a TV show called SuperMemo in ...

  3. 平衡树(Splay):Splaytree POJ 3580 SuperMemo

    SuperMemo         Description Your friend, Jackson is invited to a TV show called SuperMemo in which ...

  4. POJ 3580 SuperMemo (splay tree)

    SuperMemo Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: 6841   Accepted: 2268 Case Ti ...

  5. Splay树(多操作)——POJ 3580 SuperMemo

    相应POJ题目:点击打开链接 SuperMemo Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: 11309   Accept ...

  6. POJ 3580 SuperMemo 伸展树

    题意: 维护一个序列,支持如下几种操作: ADD x y D:将区间\([x,y]\)的数加上\(D\) REVERSE x y:翻转区间\([x,y]\) REVOLVE x y T:将区间\([x ...

  7. POJ 3580:SuperMemo(Splay)

    http://poj.org/problem?id=3580 题意:有6种操作,其中有两种之前没做过,就是Revolve操作和Min操作.Revolve一开始想着一个一个删一个一个插,觉得太暴力了,后 ...

  8. 【POJ 3580】SuperMemo Splay

    题意 给定$n$个数,$m$个询问,每次在$[L,R]$区间加上一个数,或者反转一个区间$[L,R]$,或者循环右移区间$[L,R]$共$T$次,或者在第$x$个数后插入一个数$p$,或者删除第$x$ ...

  9. 【POJ 3580】 SuperMemo

    [题目链接] 点击打开链接 [算法] 本题也是Splay区间操作的模板题,不过要比BZOJ 3223要稍微复杂一些,做完此题后,我终于对Splay有了更深入的理解,有“拨开云雾见青天”的感觉 本题还是 ...

随机推荐

  1. socket和多线程编程资料汇集-基础篇

    0 基础 CS结构的分析,server端和client的选取. 1 查看端口是否链接 netstat -an|grep portid 2 root用户抓包 tcpdump port -w fn.cap ...

  2. day4-内置函数

    一.内置函数列表 二.常见内置函数用法 由于python内置函数较多,在此总结一下部分常见的内置函数的用法: abs(x)功能:取数的绝对值 >>> abs(0) 0 >> ...

  3. Windows7+VS2008 下编译Subversion 1.8.3

    一.需要的软件包 1.python-2.7.5.msi  http://www.python.org/ 2.ActivePerl-5.8.8.822-MSWin32-x86-280952.msi  h ...

  4. atom总结

    window 系统 //查找 apm search emmet //安装 apm install emmet //删除  apm remove emmet

  5. uva1636 - Headshot(条件概率)

    简单的条件概率题,直接再来一枪没子弹的概率是所有子串”00“的数目除以‘0’的数目,随机转一下再打没子弹的概率是‘0’的数目除以总数目. #include<iostream> #inclu ...

  6. Codeforces Round #262 (Div. 2)C(二分答案,延迟标记)

    这是最大化最小值的一类问题,这类问题通常用二分法枚举答案就行了. 二分答案时,先确定答案肯定在哪个区间内.然后二分判断,关键在于怎么判断每次枚举的这个答案行不行. 我是用a[i]数组表示初始时花的高度 ...

  7. 骨骼动画 cocos2d-x + cocoStudio <cocos2d-x : version 2.2.0>

    cocos2d-x version 2.2.0 首先,在 HelloWorldScene.cpp 中的 init()函数中 添加如下代码 (资源文件可以直接到 cocos2d-x 中获取) #incl ...

  8. 创建Task的多种方法

    Gradle的Project从本质上说只是含有多个Task的容器,一个Task与Ant的Target相似,表示一个逻辑上的执行单元. 我们可以通过多种方式定义Task,所有的Task都存放在Proje ...

  9. TLD视觉跟踪算法

    TLD算法好牛逼一个,这里有个视频,是作者展示算法的效果,http://www.56.com/u83/v_NTk3Mzc1NTI.html.下面这个csdn博客里有人做的相关总结,感觉挺好的,收藏了! ...

  10. Contiki学习笔记

    http://blog.chinaunix.net/uid-9112803-id-2975824.html