记录编号 575382 评测结果 AAAAAAAAAA
题目名称 [Tyvj 1728]普通平衡树 最终得分 100
用户昵称 Gravatarlihaoze 是否通过 通过
代码语言 C++ 运行时间 0.705 s
提交时间 2022-09-13 10:50:29 内存使用 2.94 MiB
显示代码纯文本
  1. #include <bits/stdc++.h>
  2.  
  3. struct item {
  4. int key, prior, cnt, size;
  5. item *l, *r;
  6. item () { }
  7. item (int key) : key(key), prior(std::rand()), l(nullptr), r(nullptr), cnt(1), size(1) { }
  8. };
  9.  
  10. using pitem = item*;
  11.  
  12. pitem root = nullptr;
  13.  
  14. void update(pitem& x) {
  15. x->size = x->cnt + (x->l ? x->l->size : 0) + (x->r ? x->r->size : 0);
  16. }
  17.  
  18. void zig(pitem& x) {
  19. pitem y = x->l;
  20. x->l = y->r, y->r = x, x = y;
  21. update(x), update(x->r);
  22. }
  23.  
  24. void zag(pitem& x) {
  25. pitem y = x->r;
  26. x->r = y->l, y->l = x, x = y;
  27. update(x), update(x->l);
  28. }
  29.  
  30. void insert(pitem& x, int y) {
  31. if (!x)
  32. return x = new item(y), void();
  33. if (x->key == y)
  34. return ++ x->cnt, update(x), void();
  35. if (y < x->key) {
  36. insert(x->l, y);
  37. if (x->l->prior > x->prior) zig(x);
  38. } else {
  39. insert(x->r, y);
  40. if (x->r->prior > x->prior) zag(x);
  41. }
  42. update(x);
  43. }
  44.  
  45. void remove(pitem& x, int y) {
  46. if (y < x->key) remove(x->l, y);
  47. else if (y > x->key) remove(x->r, y);
  48. else {
  49. if (x->cnt > 1) -- x->cnt;
  50. else if (!x->l) x = x->r;
  51. else if (!x->r) x = x->l;
  52. else {
  53. zag(x);
  54. remove(x->l, y);
  55. if (x->l && x->l->prior > x->prior)
  56. zig(x);
  57. }
  58. }
  59. if (x) update(x);
  60. }
  61.  
  62. pitem getPre(int v) {
  63. pitem x = root, ans = new item(-1e9);
  64. while (x) {
  65. if (v == x->key)
  66. if (x->l) {
  67. x = x->l;
  68. while (x->r) x = x->r;
  69. ans = x;
  70. }
  71. if (x->key < v && x->key > ans->key)
  72. ans = x;
  73. x = v < x->key ? x->l : x->r;
  74. }
  75. return ans;
  76. }
  77.  
  78. pitem getNxt(int v) {
  79. pitem x = root, ans = new item(1e9);
  80. while (x) {
  81. if (v == x->key)
  82. if (x->r) {
  83. x = x->r;
  84. while (x->l) x = x->l;
  85. }
  86. if (x->key > v && x->key < ans->key)
  87. ans = x;
  88. x = v > x->key ? x->r : x->l;
  89. }
  90. return ans;
  91. }
  92.  
  93. int getValByRank(pitem& x, int rank) {
  94. if (!x) return 1e9;
  95. if ((x->l ? x->l->size : 0) >= rank)
  96. return getValByRank(x->l, rank);
  97. if ((x->l ? x->l->size : 0) + x->cnt >= rank)
  98. return x->key;
  99. return getValByRank(x->r, rank - (x->l ? x->l->size : 0) - x->cnt);
  100. }
  101.  
  102. int getRankByVal(pitem& x, int v) {
  103. if (!x) return 0;
  104. if (v == x->key) return (x->l ? x->l->size : 0) + 1;
  105. if (v < x->key) return getRankByVal(x->l, v);
  106. return getRankByVal(x->r, v) + (x->l ? x->l->size : 0) + x->cnt;
  107. }
  108.  
  109. int main() {
  110. freopen("phs.in", "r", stdin);
  111. freopen("phs.out", "w", stdout);
  112. root = new item(1e9), root->r = new item(-1e9);
  113. int n; std::cin >> n;
  114. while (n --) {
  115. int op, x;
  116. std::cin >> op >> x;
  117. switch (op) {
  118. case 1:
  119. insert(root, x);
  120. break;
  121. case 2:
  122. remove(root, x);
  123. break;
  124. case 3:
  125. std::cout << getRankByVal(root, x) << '\n';
  126. break;
  127. case 4:
  128. std::cout << getValByRank(root, x) << '\n';
  129. break;
  130. case 5:
  131. std::cout << getPre(x)->key << '\n';
  132. break;
  133. case 6:
  134. std::cout << getNxt(x)->key << '\n';
  135. break;
  136. }
  137. }
  138. return 0;
  139. }