bzoj 1503 平衡树(splay)

bzoj 1503 平衡树(splay)

题目简介:

   用一个数据结构来统计员工,有四种操作 1. 加入一个初始工资为A的员工 2. 将所有人工资提高一个数  3. 将所有人工资降低一个数  4. 询问第K多工资的员工是谁。 其间一点某人的工资低于工资下限,就会立刻离开公司...

吐槽:

    1. 我不是新八... 我不会吐槽...
    2. 用任何平衡树写都好.... 我是用来练splay的~~~(鄙视我这个弱菜有意思么)
    3. 为毛刚进公司就离开的人不需要统计.... 题目里面也没说啊... 害得我查了一晚上... 最后在插入的时候加个判断就过了...

思路解析:

    插入和查询就是splay的基本操作了~
    删除半区间我是把(工资下限-1)插入到splay当中再旋到树根(splay经典操作)然后删除左子树和树根。
    工资的升降可以直接更改工资下限,将更改的总量记录一下~
    
    splay可以只写zig和zig-zig操作,zig-zag可以归约掉... Orzzzzzz...... 一下子好写了很多
    维护size域我是从下到上维护的...因为需要更新size的只有x的直系祖先,所以旋到树根的过程相当于更新了...
    我终于会写splay了... 好高欣~
   
   1 #include<iostream>
  2 #include<cstdio>
  3 #include<cstdlib>
  4  using  namespace std;
  5  #define re(i,n) for(int i =0; i< n; i++)
  6  const  int N = 100005;
  7  const  int inf = ~0u>>2;
  8  int sum,splaysz,temp;
  9  //  template 
 10  struct node {
 11      int p,chd[2],sz,mul,v;
 12     node( int P=0, int val=0){
 13         chd[0] = chd[1] = 0; sz = 0; mul = 1;
 14         p = P, v = val;
 15     }
 16 } tree[N];
 17 inline  void sc( int y,  int x,  int p){tree[y].chd[p] = x, tree[x].p = y;}
 18 inline  void upd( int x){ tree[x].sz = tree[tree[x].chd[0]].sz + tree[tree[x].chd[1]].sz+tree[x].mul;}
 19 inline  void rot( int x){
 20      int y = tree[x].p;
 21      int w = tree[y].chd[1] == x;
 22     sc(tree[y].p,x,tree[tree[y].p].chd[1] == y);
 23     sc(y,tree[x].chd[w^1], w);
 24     sc(x,y,w^1);
 25     upd(y); upd(x);
 26 }
 27  void splay( int x, int rt){
 28      while(tree[x].p != rt){
 29          int y = tree[x].p;
 30          int w = tree[y].chd[1] == x;
 31          if(tree[y].p != rt && tree[tree[y].p].chd[w] == y) rot(y);
 32         rot(x);
 33     }
 34 }
 35  //  operate 
 36  void ins( int val){
 37      if(tree[0].chd[0] == 0) {tree[++splaysz] = node(0,val); tree[0].chd[0] = splaysz; upd(splaysz);  return;}
 38      int now = tree[0].chd[0];
 39      while(1){
 40          if(val == tree[now].v){
 41             tree[now].mul++;
 42             upd(now);
 43             splay(now,0);
 44              return ;
 45         }
 46          int w = val > tree[now].v;
 47          if(tree[now].chd[w]) now = tree[now].chd[w];
 48          else {
 49             tree[++splaysz] = node(now,val); tree[now].chd[w] = splaysz;
 50             upd(splaysz);
 51             splay(splaysz,0);
 52              return ;
 53         }
 54     }
 55 }
 56  void find( int pos, int k){
 57      if (tree[pos].sz < k) {
 58         puts("-1");
 59          return ;
 60     }
 61      int x = tree[pos].chd[0];
 62      if(tree[x].sz>=k) find(x,k);
 63      else  if(tree[x].sz + tree[pos].mul>=k) {
 64         printf("%d\n",tree[pos].v-temp);
 65     }
 66      else find(tree[pos].chd[1], k - tree[x].sz - tree[pos].mul);
 67 }
 68  void del(){
 69      int r = tree[0].chd[0];
 70      int x = tree[r].chd[1];
 71     sum += tree[r].sz - tree[x].sz -1;
 72     tree[x].p = 0 , tree[0].chd[0] = x;
 73 }
 74  //  main
 75  int main(){
 76      int n,m,x;
 77      char ch[10];
 78      while(cin >>n >>m){
 79         sum = temp = splaysz = 0;
 80         tree[0] = node(0,inf);
 81         re(i,n){
 82             scanf("%s%d",ch,&x);
 83              switch(ch[0]){
 84                  case 'I' :
 85                      if(x >= m)
 86                         ins(x + temp);
 87                      break;
 88                  case 'S' :
 89                     temp += x;
 90                     ins(m + temp-1); del();
 91                      break;
 92                  case 'A' :
 93                     temp -= x;
 94                      break;
 95                  case 'F' :
 96                      if(tree[tree[0].chd[0]].sz < x) puts("-1");
 97                      else find(tree[0].chd[0],tree[tree[0].chd[0]].sz - x + 1);
 98                      break;
 99             }
100         }
101         printf("%d\n",sum);
102     }
103 }
104 

你可能感兴趣的:(bzoj 1503 平衡树(splay))