传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=1014

天,写kth()时,把判断条件k <= siz[ch[x][0]]错写成了k <= ch[x][0],RE不停,还爆掉了几个小时,以后写数据结构题一定要头脑清晰啊!

  1. #include <cstdio>
  2. #include <cstring>
  3.  
  4. const int maxn = 300005, base = 131;
  5.  
  6. int fa[maxn], ch[maxn][2], key[maxn], siz[maxn], root, cnt;
  7. unsigned long long hash[maxn], poww[maxn];
  8. int n, m, t1, t2, stk[maxn], top;
  9. char s[maxn], opr, t3;
  10.  
  11. inline void pushup(int x) {
  12. hash[x] = hash[ch[x][0]] * (poww[siz[ch[x][1]] + 1]) + key[x] * poww[siz[ch[x][1]]] + hash[ch[x][1]];
  13. siz[x] = siz[ch[x][0]] + siz[ch[x][1]] + 1;
  14. }
  15. inline void rotate(int x) {
  16. int y = fa[x];
  17. if (y == ch[fa[y]][0]) {
  18. ch[fa[y]][0] = x;
  19. }
  20. else {
  21. ch[fa[y]][1] = x;
  22. }
  23. fa[x] = fa[y];
  24. int dir = x == ch[y][1];
  25. ch[y][dir] = ch[x][dir ^ 1];
  26. fa[ch[x][dir ^ 1]] = y;
  27. ch[x][dir ^ 1] = y;
  28. fa[y] = x;
  29. pushup(y);
  30. pushup(x);
  31. }
  32. inline void splay(int x, int rt) {
  33. int p;
  34. while (fa[x] != rt) {
  35. p = fa[x];
  36. if (fa[p] == rt) {
  37. rotate(x);
  38. }
  39. else {
  40. if ((p == ch[fa[p]][1]) ^ (x == ch[p][1])) {
  41. rotate(x);
  42. }
  43. else {
  44. rotate(p);
  45. }
  46. rotate(x);
  47. }
  48. }
  49. if (!rt) {
  50. root = x;
  51. }
  52. }
  53. int make_tree(int left, int right) {
  54. if (left > right) {
  55. return 0;
  56. }
  57. int rt = (left + right) >> 1;
  58. key[rt] = s[rt];
  59. ch[rt][0] = make_tree(left, rt - 1);
  60. fa[ch[rt][0]] = rt;
  61. ch[rt][1] = make_tree(rt + 1, right);
  62. fa[ch[rt][1]] = rt;
  63. pushup(rt);
  64. return rt;
  65. }
  66. inline int kth(int k) {
  67. int x = root;
  68. while (k != siz[ch[x][0]] + 1) {
  69. if (k <= siz[ch[x][0]]) {
  70. x = ch[x][0];
  71. }
  72. else {
  73. k -= siz[ch[x][0]] + 1;
  74. x = ch[x][1];
  75. }
  76. }
  77. return x;
  78. }
  79. inline void modify(int root1, int root2) {
  80. root1 = kth(root1);
  81. root2 = kth(root2);
  82. splay(root1, 0);
  83. splay(root2, root1);
  84. }
  85. inline void ist(int pos, int _key) {
  86. modify(pos, pos + 1);
  87. ++cnt;
  88. hash[cnt] = key[cnt] = _key;
  89. siz[cnt] = 1;
  90. ch[ch[root][1]][0] = cnt;
  91. fa[cnt] = ch[root][1];
  92. pushup(ch[root][1]);
  93. pushup(root);
  94. }
  95. /*void ist(int x, int k, int _key) {
  96. if (k == siz[ch[x][0]] + 1) {
  97. if (ch[x][1]) {
  98. top = 0;
  99. int i;
  100. for (i = ch[x][1]; ch[i][0]; i = ch[i][0]) {
  101. stk[top++] = i;
  102. }
  103. ++cnt;
  104. ch[i][0] = cnt;
  105. fa[cnt] = i;
  106. key[cnt] = _key;
  107. siz[cnt] = 1;
  108. pushup(i);
  109. for (i = top - 1; ~i; --i) {
  110. pushup(stk[i]);
  111. }
  112. }
  113. else {
  114. ++cnt;
  115. ch[x][1] = cnt;
  116. fa[cnt] = x;
  117. key[cnt] = _key;
  118. }
  119. pushup(x);
  120. splay(cnt, 0);
  121. return;
  122. }
  123. if (k <= siz[ch[x][0]]) {
  124. ist(ch[x][0], k, _key);
  125. }
  126. else {
  127. ist(ch[x][1], k - siz[ch[x][0]] - 1, _key);
  128. }
  129. }*/
  130. void upd(int x, int k, int _key) {
  131. if (k == siz[ch[x][0]] + 1) {
  132. key[x] = _key;
  133. pushup(x);
  134. return;
  135. }
  136. if (k <= siz[ch[x][0]]) {
  137. upd(ch[x][0], k, _key);
  138. }
  139. else {
  140. upd(ch[x][1], k - siz[ch[x][0]] - 1, _key);
  141. }
  142. pushup(x);
  143. }
  144. unsigned long long gethash(int start, int end) {
  145. modify(start - 1, end + 1);
  146. return hash[ch[ch[root][1]][0]];
  147. }
  148.  
  149. int main(void) {
  150. //freopen("in.txt", "r", stdin);
  151. //freopen("out.txt", "w", stdout);
  152. scanf("%s", s + 2);
  153. n = strlen(s + 2);
  154. s[1] = s[n + 2] = '$';
  155. n += 2;
  156. cnt = n;
  157. poww[0] = 1;
  158. for (int i = 1; i < maxn; ++i) {
  159. poww[i] = poww[i - 1] * base;
  160. }
  161. root = make_tree(1, n);
  162. scanf("%d", &m);
  163. int left, right, mid;
  164. while (m--) {
  165. while ((opr = getchar()) < 'A');
  166. scanf("%d", &t1);
  167. ++t1;
  168. if (opr == 'I') {
  169. while ((t3 = getchar()) < 'a');
  170. ist(t1, t3);
  171. ++n;
  172. }
  173. else if (opr == 'R') {
  174. while ((t3 = getchar()) < 'a');
  175. upd(root, t1, t3);
  176. }
  177. else {
  178. scanf("%d", &t2);
  179. ++t2;
  180. left = 0;
  181. right = n - (t1 > t2? t1: t2);
  182. while (left != right) {
  183. mid = (left + right + 1) >> 1;
  184. if (gethash(t1, t1 + mid - 1) == gethash(t2, t2 + mid - 1)) {
  185. left = mid;
  186. }
  187. else {
  188. right = mid - 1;
  189. }
  190. }
  191. printf("%d\n", left);
  192. }
  193. }
  194. return 0;
  195. }

  

_bzoj1014 [JSOI2008]火星人prefix【Splay】的更多相关文章

  1. BZOJ 1014: [JSOI2008]火星人prefix [splay 二分+hash] 【未完】

    1014: [JSOI2008]火星人prefix Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 6243  Solved: 2007[Submit] ...

  2. BZOJ 1014: [JSOI2008]火星人prefix Splay+二分

    1014: [JSOI2008]火星人prefix 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=1014 Description 火星人 ...

  3. BZOJ 1014: [JSOI2008]火星人prefix( splay + hash )

    用splay维护序列, 二分+hash来判断LCQ.. #include<bits/stdc++.h> using namespace std; typedef unsigned long ...

  4. BZOJ 1014 [JSOI2008]火星人prefix (Splay + Hash + 二分)

    1014: [JSOI2008]火星人prefix Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 8112  Solved: 2569[Submit] ...

  5. 【BZOJ1014】[JSOI2008]火星人prefix Splay+hash

    [BZOJ1014][JSOI2008]火星人prefix Description 火星人最近研究了一种操作:求一个字串两个后缀的公共前缀.比方说,有这样一个字符串:madamimadam,我们将这个 ...

  6. [BZOJ1014] [JSOI2008] 火星人prefix (splay & 二分答案)

    Description 火星人最近研究了一种操作:求一个字串两个后缀的公共前缀.比方说,有这样一个字符串:madamimadam,我们将这个字符串的各个字符予以标号:序号: 1 2 3 4 5 6 7 ...

  7. bzoj1014: [JSOI2008]火星人prefix splay+hash+二分

    Description 火星人最近研究了一种操作:求一个字串两个后缀的公共前缀.比方说,有这样一个字符串:madamimadam,我们将这个字符串的各个字符予以标号:序号: 1 2 3 4 5 6 7 ...

  8. [bzoj1014](JSOI2008)火星人 prefix (Splay维护哈希)

    Description 火星人最近研究了一种操作:求一个字串两个后缀的公共前缀. 比方说,有这样一个字符串:madamimadam,我们将这个字符串的各个字符予以标号:序号: 1 2 3 4 5 6 ...

  9. 【bzoj1014】[JSOI2008]火星人prefix Splay+Hash+二分

    题目描述 火星人最近研究了一种操作:求一个字串两个后缀的公共前缀.比方说,有这样一个字符串:madamimadam,我们将这个字符串的各个字符予以标号:序号: 1 2 3 4 5 6 7 8 9 10 ...

随机推荐

  1. HDU 1588 Gauss Fibonacci(矩阵高速幂+二分等比序列求和)

    HDU 1588 Gauss Fibonacci(矩阵高速幂+二分等比序列求和) ACM 题目地址:HDU 1588 Gauss Fibonacci 题意:  g(i)=k*i+b;i为变量.  给出 ...

  2. C#:excel导入导出

    资源:excelService 服务 http://download.csdn.net/detail/istend/8060501 排列问题 导出时,数字和字符的排列格式默认不一样,数字靠右,字符靠左 ...

  3. 【转】Web Worker javascript多线程编程(一)

    原文:https://www.cnblogs.com/peakleo/p/6218823.html -------------------------------------------------- ...

  4. Java知识图谱

    1.Java学习路径1 我想很多人看到这个路径可能会问我在哪里可以学习,所以就先附上这条路径的学习地址吧,这也是这张图片的来源,愿意学习的可以去看看:Java研发工程师学习路径 2.Java学习路径2 ...

  5. Solr 文章集成

    Solr 文章集成 solr原理 solr wiki: http://wiki.apache.org/solr/ 分布式全文检索系统SolrCloud简单介绍 http://my.oschina.ne ...

  6. Linux Shell_test

    test: 测试Shell脚本里的条件,通过推出状态返回其结果.用法:    test [ expression ] 或 [ [ expression ] ]    注意空格test表达式:是则为真  ...

  7. Android 封装实现各种样式对话框

    先上图 watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/disso ...

  8. 两个喜欢的"新"C#语法

    现在C#比较新的语法,我都十分喜欢. 比如属性可设默认值: public string Name { get; set; } = "张三"; 还有一个就是拼接字符串. 以往,通常都 ...

  9. bzoj3272: Zgg吃东西&&3267: KC采花

    口胡 我们容易得到一个费用流的做法,流出k的流量分配给各个点,各个点向下一个点流费用为它的价值的边,然后汇总到ed 观察发现对于流一次,相当于选择了一个区间 如果流了反向边,相当于减去了这一段 可以用 ...

  10. HDU1069 Monkey and Banana —— DP

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=1069 Monkey and Banana Time Limit: 2000/1000 MS ...