解法
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); } } }
0 件のコメント:
コメントを投稿