P2596 [ZJOI2006]书架(fhq treap)

P2596 [ZJOI2006]书架

我们用fhq treap来完成这一题

对于一个新插入的节点我们取权值为其索引值,其所记录的 v a l u e value value是其当前索引所在位置。

操作一:把索引为 v a l u e value value的点放到平衡树前面,分别别得到三颗子树 x , y , z x, y, z x,y,z(前段子树,索引为 v a l u e value value所代表的子树,后段子树),同时把 v a l [ y ] val[y] val[y]修改成全局最小,然后按照顺序 m e r g e ( y , x , z ) merge(y, x, z) merge(y,x,z)

操作二:与操作一类似得到 x , y , z x, y, z x,y,z,然后把 v a l [ y ] val[y] val[y]改成全局最大,按照顺序 m e r g e ( x , z , y ) merge(x, z, y) merge(x,z,y)

操作三: t = 0 t = 0 t=0不做操作, t = 1 t = 1 t=1得到四颗子树 x , y , z , w x, y, z, w x,y,z,w(前段子树, v a l u e value value所代表的子树, v a l u e value value的后继,后端子树),交换信息,然后按照顺序 m e r g e ( x , z , y , w ) merge(x, z, y, w) merge(x,z,y,w) t = − 1 t = -1 t=1得到四颗子树 x , y , z , w x, y, z, w x,y,z,w(前段子树, v a l u e value value的前驱, v a l u e value value所代表的子树,后端子树),交换信息,然后按照顺序 m e r g e ( x , z , y , w ) merge(x, z, y, w) merge(x,z,y,w)

操作四:按照值分裂成两棵树,然后输出第一颗树的大小即可。

操作五:直接查找第 k k k大即可。

#include <bits/stdc++.h>

using namespace std;

const int N = 1e5 + 10;

mt19937 rnd(233);

int minn, maxn;

struct Tree {
  int ls[N], rs[N], val[N], sz[N], key[N], root, cnt;

  
  void push_up(int rt) {
    sz[rt] = sz[ls[rt]] + sz[rs[rt]] + 1;
  }

  int new_node(int value, int pos) {
    val[value] = pos, sz[value] = 1, key[value] = rnd();
    return value;
  }

  void split(int rt, int value, int &x, int &y) {
    if (!rt) {
      x = y = 0;
      return ;
    }
    if (val[rt] <= value) {
      x = rt;
      split(rs[rt], value, rs[rt], y);
    }
    else {
      y = rt;
      split(ls[rt], value, x, ls[rt]);
    }
    push_up(rt);
  }

  int merge(int x, int y) {
    if (!x || !y) {
      return x | y;
    }
    if (key[x] < key[y]) {
      rs[x] = merge(rs[x], y);
      push_up(x);
      return x;
    }
    else {
      ls[y] = merge(x, ls[y]);
      push_up(y);
      return y;
    }
  }

  int get_num(int rt, int rank) {
    while (rt) {
      if (sz[ls[rt]] + 1 == rank) {
        break;
      }
      if (sz[ls[rt]] >= rank) {
        rt = ls[rt];
      }
      else {
        rank -= sz[ls[rt]] + 1;
        rt = rs[rt];
      }
    }
    return rt;
  }

  int get_num(int rank) {
    return get_num(root, rank);
  }

  void insert(int value, int pos) {
    int x, y;
    split(root, pos, x, y);
    root = merge(merge(x, new_node(value, pos)), y);
  }

  void update(int value, int op) {
    int x, y, z;
    split(root, val[value], x, z);
    split(x, val[value] - 1, x, y);
    if (op) {
      val[value] = --minn;
      root = merge(merge(y, x), z);
    }
    else {
      val[value] = ++maxn;
      root = merge(merge(x, z), y);
    }
  }

  void reverse(int value, int op) {
    if (!op) {
      return ;
    }
    if (op == 1) {
      int x, y, z, w;
      split(root, val[value], x, z);
      split(x, val[value] - 1, x, y);
      int t = get_num(z, 1);
      split(z, val[t], z, w);
      swap(val[y], val[z]);
      root = merge(merge(x, z), merge(y, w));
    }
    else {
      int x, y, z, w;
      split(root, val[value] - 1, x, z);
      split(z, val[value], z, w);
      int t = get_num(x, sz[x]);
      split(x, val[t] - 1, x, y);
      swap(val[y], val[z]);
      root = merge(merge(x, z), merge(y, w));
    }
  }

  int get_rank(int value) {
    int x, y, ans;
    split(root, val[value] - 1, x, y);
    ans = sz[x];
    root = merge(x, y);
    return ans;
  }
}tree;

int main() {
  // freopen("in.txt", "r", stdin);
  // freopen("out.txt", "w", stdout);
  // ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
  int n, m;
  scanf("%d %d", &n, &m);
  minn = 1, maxn = n;
  for (int i = 1, x; i <= n; i++) {
    scanf("%d", &x);
    tree.insert(x, i);
  }
  char op[10];
  for (int i = 1, s, t; i <= m; i++) {
    scanf("%s %d", op, &s);
    if (op[0] == 'T') {
      tree.update(s, 1);
    }
    else if (op[0] == 'B') {
      tree.update(s, 0);
    }
    else if (op[0] == 'I') {
      scanf("%d", &t);
      tree.reverse(s, t);
    }
    else if (op[0] == 'A') {
      printf("%d\n", tree.get_rank(s));
    }
    else {
      printf("%d\n", tree.get_num(s));
    }
  }
  return 0;
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
可持久化splay是一种数据结构,它是对splay树进行修改和查询的一种扩展。在传统的splay树中,对树的修改操作会破坏原有的树结构,而可持久化splay树则允许我们对树进行修改、查询,并且可以保存修改后的每个版本的树结构。 在可持久化splay树中,我们不会直接对原树进行修改,而是通过复制每个节点来创建新的版本。这样,每个版本都可以独立地修改和查询,保留了原有版本的结构和状态。每个节点保存了其左子树和右子树的引用,使得可以在不破坏原有版本的情况下进行修改和查询。 为了实现可持久化splay树,我们可以使用一些技巧,比如引用中提到的哨兵节点和假的父节点和孩子节点。这些技巧可以帮助我们处理根节点的旋转和其他操作。 此外,可持久化splay树还可以与其他数据结构相结合,比如引用中提到的可持久化线段树。这种结合可以帮助我们解决更复杂的问题,比如区间修改和区间查询等。 对于可持久化splay树的学习过程,可以按照以下步骤进行: 1. 理解splay树的基本原理和操作,包括旋转、插入、删除和查找等。 2. 学习如何构建可持久化splay树,包括复制节点、更新版本和保存历史版本等。 3. 掌握可持久化splay树的常见应用场景,比如区间修改和区间查询等。 4. 深入了解与可持久化splay树相关的其他数据结构和算法,比如可持久化线段树等。 在解决问题时,可以使用二分法来确定答案,一般称为二分答案。通过对答案进行二分,然后对每个答案进行检查,以确定最终的结果。这种方法可以应用于很多问题,比如引用中提到的在线询问问题。 综上所述,可持久化splay是一种对splay树进行修改和查询的扩展,可以通过复制节点来创建新的版本,并且可以与其他数据结构相结合解决更复杂的问题。学习过程中可以按照一定的步骤进行,并且可以使用二分法来解决一些特定的问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [[学习笔记]FHQ-Treap及其可持久化](https://blog.csdn.net/weixin_34283445/article/details/93207491)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [可持久化数据结构学习笔记](https://blog.csdn.net/weixin_30376083/article/details/99902410)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值