解法
Treap木を本とかを大分参考にしながら実装してみた。
割と実装楽しい。
ソース
#include<bits/stdc++.h>
using namespace std;
const int INF = 1 << 30;
struct Treap {
struct node{
int value; // 値
node *left, *right; // *左, *右:
int priority; // 優先度
int cnt; // 部分木のサイズ
// int sum; // 部分木の和
int small; //部分木の最小値
node( int v) : value(v), priority(rand()), cnt(1), /*sum(v),*/ small(v){
left = right = NULL;
}
};
node *root;
Treap():root(NULL){}
int count( node *t){ return !t ? 0 : t->cnt; }
//int sum( node *t) { return !t ? 0 : t->sum; }
int mini( node *t) { return !t ? INF : t->small; }
node* update( node* t) {
t -> cnt = count( t -> left) + count( t -> right) + 1;
// t -> sum = sum( t -> left) + sum( t -> right) + t -> value;
t -> small = min( min( mini( t -> left), mini( t -> right)), t -> value);
return t;
}
node* merge( node* l, node* r){
if(!l || !r) return l? l : r;
if(l -> priority > r -> priority) {
l -> right = merge( l -> right, r);
return update(l);
} else {
r -> left = merge( l, r -> left);
return update(r);
}
}
pair< node*, node* > split( node* t, int k){ // [0,k),[k,n)
if(!t) return make_pair((node*)NULL,(node*)NULL);
if(k <= count( t -> left)){
pair< node*, node* > s = split( t -> left, k);
t -> left = s.second;
return make_pair( s.first, update(t));
}else{
pair< node*, node* > s = split( t -> right, k - count( t -> left) - 1);
t -> right = s.first;
return make_pair( update(t), s.second);
}
}
node* insert( node* t, int k, int val){
pair< node*, node* > s = split( t, k);
t = merge( s.first, new node( val));
return update(merge( t, s.second));
}
node* erase( node* t, int k){
pair< node*, node* > s1, s2;
s1 = split( t, k + 1);
s2 = split( s1.first, k);
t = merge( s2.first, s1.second);
return update(t);
}
node* find( node* t, int k){
int c = count(t -> left);
if(k < c) return find( t -> left, k);
else if(k > c) return find( t -> right, k - c - 1);
else return t;
}
void insert( int k, int v){ root = insert( root, k, v);}
void erase( int k) { root = erase( root, k);}
node* find( int k) { return find( root, k); }
};
int main(){
int n, q;
Treap tree;
scanf("%d %d", &n, &q);
for(int i = 0, a ; i < n ; i++ ){
scanf("%d", &a);
tree.insert( i, a);
}
while(q--){
int x, y, z;
scanf("%d %d %d", &x, &y, &z);
if(x == 0){
z++;
pair< Treap::node*, Treap::node* > a, b, c;
c = tree.split( tree.root, z); // [0,z] [z + 1,n)
b = tree.split( c.first , z - 1); // [0,z) [z]
a = tree.split( b.first , y); // [0,y) [y,z)
tree.root = tree.merge( a.first, b.second); // [0,y) + [z]
tree.root = tree.merge( tree.root, a.second); // root + [y,z)
tree.root = tree.merge( tree.root, c.second); // root + [z+1,n)
//ふええふえぇふぇぇ(むずかしいね
} else if(x == 1){
z++;
pair< Treap::node*, Treap::node* > p, q;
p = tree.split( tree.root, y);
q = tree.split( p.second, z - y);
printf("%d\n", tree.mini(q.first));
tree.root = tree.merge( p.first, tree.merge( q.first, q.second));
} else {
tree.erase(y);
tree.insert( y, z);
}
}
}