求树上两条路径的 LCP (树上每个节点代表一个字符)

总共写+调了6个多小时,终于过了~

绝对是我写过的最复杂的数据结构了

我们对这棵树进行轻重链剖分,然后把所有的重链分正串,反串插入到广义后缀自动机中.

求 LCP 的话就是后缀树中两点 $LCA$ 的深度.

如果 $LCP$ 的长度小于两个重链的长度,就直接输出答案,否则还要继续爬重链.

这道题恶心之处在于将所有重链不重不漏存起来,无法形容有多恶心.

code:

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <vector>
  4. #include <queue>
  5. #include <algorithm>
  6. #include <stack>
  7. #include <string>
  8. using namespace std;
  9. void setIO(string s)
  10. {
  11. string in=s+".in";
  12. string out=s+".out";
  13. freopen(in.c_str(),"r",stdin);
  14. freopen(out.c_str(),"w",stdout);
  15. }
  16. namespace sam
  17. {
  18. #define N 1200004
  19. int last,tot;
  20. int ch[N][27],pre[N],mx[N];
  21. inline int extend(int c)
  22. {
  23. if(ch[last][c])
  24. {
  25. int p=last;
  26. int q=ch[p][c];
  27. if(mx[q]==mx[p]+1) last=q;
  28. else
  29. {
  30. int nq=++tot;
  31. mx[nq]=mx[p]+1;
  32. memcpy(ch[nq],ch[q],sizeof(ch[q]));
  33. pre[nq]=pre[q],pre[q]=nq;
  34. for(;p&&ch[p][c]==q;p=pre[p]) ch[p][c]=nq;
  35. last=nq;
  36. }
  37. }
  38. else
  39. {
  40. int np=++tot,p=last;
  41. mx[np]=mx[p]+1,last=np;
  42. for(;p&&!ch[p][c];p=pre[p]) ch[p][c]=np;
  43. if(!p) pre[np]=1;
  44. else
  45. {
  46. int q=ch[p][c];
  47. if(mx[q]==mx[p]+1) pre[np]=q;
  48. else
  49. {
  50. int nq=++tot;
  51. mx[nq]=mx[p]+1;
  52. memcpy(ch[nq],ch[q],sizeof(ch[q]));
  53. pre[nq]=pre[q],pre[np]=pre[q]=nq;
  54. for(;p&&ch[p][c]==q;p=pre[p]) ch[p][c]=nq;
  55. }
  56. }
  57. last=np;
  58. }
  59. return last;
  60. }
  61. #undef N
  62. };
  63.  
  64. // 优美的后缀自动机
  65.  
  66. namespace suffix_tree
  67. {
  68. #define LOG 21
  69. #define N 1200006
  70. vector<int>G[N];
  71. int fa[LOG][N],dis[N];
  72. void dfs(int u)
  73. {
  74. int i,j;
  75. dis[u]=dis[sam::pre[u]]+1;
  76. fa[0][u]=sam::pre[u];
  77. for(i=1;i<LOG;++i) fa[i][u]=fa[i-1][fa[i-1][u]];
  78. for(i=0;i<(int)G[u].size();++i) dfs(G[u][i]);
  79. }
  80. inline void build()
  81. {
  82. int i,j;
  83. for(i=2;i<=sam::tot;++i)
  84. G[sam::pre[i]].push_back(i);
  85. dfs(1);
  86. }
  87. inline int jump(int x,int len)
  88. {
  89. for(int i=LOG-1;i>=0;--i) if(sam::mx[fa[i][x]]>=len) x=fa[i][x];
  90. return x;
  91. }
  92. inline int LCA(int x,int y)
  93. {
  94. if(dis[x]!=dis[y])
  95. {
  96. if(dis[x]>dis[y]) swap(x,y);
  97. for(int i=LOG-1;i>=0;--i) if(dis[fa[i][y]]>=dis[x]) y=fa[i][y];
  98. }
  99. if(x==y) return x;
  100. for(int i=LOG-1;i>=0;--i) if(fa[i][x]!=fa[i][y]) x=fa[i][x], y=fa[i][y];
  101. return fa[0][x];
  102. }
  103. #undef N
  104. #undef LOG
  105. };
  106.  
  107. // 优美的后缀树
  108.  
  109. namespace tree
  110. {
  111. #define N 600003
  112. int tot,iii;
  113. int son[N],size[N],dep[N],dfn[N],top[N],fa[N],up[N],down[N],val[N],idx[N],pp[N];
  114. vector<int>G[N];
  115. struct node
  116. {
  117. int u,c;
  118. node(int u=0,int c=0):u(u),c(c){}
  119. };
  120. vector<node>A[N];
  121. inline void add(int u,int v)
  122. {
  123. G[u].push_back(v);
  124. G[v].push_back(u);
  125. }
  126. void dfs1(int u,int ff)
  127. {
  128. fa[u]=ff,dep[u]=dep[ff]+1,size[u]=1;
  129. for(int i=0;i<G[u].size();++i)
  130. {
  131. if(G[u][i]==ff) continue;
  132. dfs1(G[u][i],u);
  133. size[u]+=size[G[u][i]];
  134. if(size[G[u][i]]>size[son[u]]) son[u]=G[u][i];
  135. }
  136. }
  137. void dfs2(int u,int tp)
  138. {
  139. idx[u]=++iii;
  140. pp[iii]=u;
  141. top[u]=tp;
  142. if(tp==u) dfn[tp]=++tot;
  143. A[dfn[tp]].push_back(node(u,val[u]));
  144. if(son[u]) dfs2(son[u],tp);
  145. for(int i=0;i<G[u].size();++i)
  146. if(G[u][i]!=son[u]&&G[u][i]!=fa[u])
  147. dfs2(G[u][i], G[u][i]);
  148. }
  149. inline void build()
  150. {
  151. int i,j;
  152. dfs1(1,0);
  153. dfs2(1,1);
  154. for(i=1;i<=tot;++i)
  155. {
  156. sam::last=1;
  157. // 从上到下 (实际是从下到上)
  158. for(j=0;j<A[i].size();++j)
  159. {
  160. int u=A[i][j].u;
  161. int c=A[i][j].c;
  162. up[u]=sam::extend(c);
  163. }
  164. // 从下到上 (实际上是从上到下)
  165. sam::last=1;
  166. for(j=A[i].size()-1;j>=0;--j)
  167. {
  168. int u=A[i][j].u;
  169. int c=A[i][j].c;
  170. down[u]=sam::extend(c);
  171. }
  172. }
  173. }
  174. int LCA(int x,int y)
  175. {
  176. while(top[x]!=top[y]) dep[top[x]]>dep[top[y]]?x=fa[top[x]]:y=fa[top[y]];
  177. return dep[x]<dep[y]?x:y;
  178. }
  179. #undef N
  180. };
  181.  
  182. // 优美的树链剖分
  183.  
  184. struct node
  185. {
  186. int u,len,ty,tt;
  187. node(int u=0,int len=0,int ty=0,int tt=0):u(u),len(len),ty(ty),tt(tt){}
  188. };
  189. #define N 300003
  190. char S[N];
  191. int ty;
  192. vector<node>A[2];
  193. stack<node>ss;
  194. void insert(int a,int b,int id)
  195. {
  196. int lca=tree::LCA(a,b);
  197. // 向上走
  198. while(tree::dep[a]>tree::dep[lca])
  199. {
  200. int pp=tree::dep[tree::top[a]]<tree::dep[lca]?lca:tree::top[a];
  201. A[id].push_back(node(tree::up[a], tree::dep[a]-max(tree::dep[lca],tree::dep[tree::top[a]])+1,0,a));
  202. a=tree::fa[tree::top[a]];
  203. }
  204. // 向下走
  205. if(a==lca)
  206. {
  207. // 未计算 lca
  208. while(tree::dep[b]>=tree::dep[lca])
  209. {
  210. int pp=tree::dep[tree::top[b]]<tree::dep[lca]?lca:tree::top[b];
  211. ss.push(node(tree::down[pp], tree::dep[b]-max(tree::dep[lca],tree::dep[pp])+1,1,pp));
  212. b=tree::fa[tree::top[b]];
  213. }
  214. }
  215. else
  216. {
  217. while(tree::dep[b]>tree::dep[lca])
  218. {
  219. int pp=tree::dep[tree::top[b]]<tree::dep[lca]?lca:tree::top[b];
  220. ss.push(node(tree::down[pp], tree::dep[b]-max(tree::dep[lca],tree::dep[pp])+1,1,pp));
  221. b=tree::fa[tree::top[b]];
  222. }
  223. }
  224. while(!ss.empty()) A[id].push_back(ss.top()),ss.pop();
  225. }
  226. int main()
  227. {
  228. // setIO("input");
  229. int i,j,n,m;
  230. scanf("%d%s",&n,S+1);
  231. for(i=1;i<=n;++i) tree::val[i]=S[i]-'a';
  232. for(i=1;i<n;++i)
  233. {
  234. int u,v;
  235. scanf("%d%d",&u,&v);
  236. tree::add(u,v);
  237. }
  238. sam::tot=1;
  239. tree::build(); // 建完树剖 + 上/下走定位到后缀自动机的位置
  240. suffix_tree::build(); // 建完后缀树
  241. scanf("%d",&m);
  242. for(i=1;i<=m;++i)
  243. {
  244. int a,b,c,d;
  245. scanf("%d%d%d%d",&a,&b,&c,&d);
  246. insert(a,b,0);
  247. insert(c,d,1);
  248. int t1=0,t2=0,lcp=0;
  249. while(t1<A[0].size()&&t2<A[1].size())
  250. {
  251. node a1=A[0][t1], a2=A[1][t2];
  252. int lca;
  253. int uu=a1.u;
  254. int vv=a2.u;
  255. int u=a1.tt;
  256. int v=a2.tt;
  257. int tmp=min(sam::mx[lca=suffix_tree::LCA(uu,vv)], min(a1.len,a2.len));
  258. lcp+=tmp;
  259. if(tmp<min(a1.len,a2.len)) break;
  260. else
  261. {
  262. if(tmp==a1.len) ++t1;
  263. else
  264. {
  265. A[0][t1].u=(A[0][t1].ty==1)?(tree::down[tree::pp[tree::idx[u]+tmp]]):(tree::up[tree::pp[tree::idx[u]-tmp]]);
  266. A[0][t1].tt=(A[0][t1].ty==1)?(tree::pp[tree::idx[u]+tmp]):(tree::pp[tree::idx[u]-tmp]);
  267. A[0][t1].len=A[0][t1].len-tmp;
  268. }
  269. if(tmp==a2.len) ++t2;
  270. else
  271. {
  272. A[1][t2].u=(A[1][t2].ty==1)?(tree::down[tree::pp[tree::idx[v]+tmp]]):(tree::up[tree::pp[tree::idx[v]-tmp]]);
  273. A[1][t2].tt=(A[1][t2].ty==1)?(tree::pp[tree::idx[v]+tmp]):(tree::pp[tree::idx[v]-tmp]);
  274. A[1][t2].len=A[1][t2].len-tmp;
  275. }
  276. }
  277. }
  278. A[0].clear(),A[1].clear();
  279. printf("%d\n",lcp);
  280. }
  281. return 0;
  282. }

  

CF504E Misha and LCP on Tree 后缀自动机+树链剖分+倍增的更多相关文章

  1. CF 504E Misha and LCP on Tree——后缀数组+树链剖分

    题目:http://codeforces.com/contest/504/problem/E 树链剖分,把重链都接起来,且把每条重链的另一种方向的也都接上,在这个 2*n 的序列上跑后缀数组. 对于询 ...

  2. CF G. Indie Album 广义后缀自动机+树链剖分+线段树合并

    这里给出一个后缀自动机的做法. 假设每次询问 $t$ 在所有 $s$ 中的出现次数,那么这是非常简单的: 直接对 $s$ 构建后缀自动机,随便维护一下 $endpos$ 大小就可以. 然而,想求 $t ...

  3. [LuoguU41039]PION后缀自动机 树链剖分+动态开点线段树

    链接 刚开始看出题人题解都吓蒙掉了,还以为是什么难题,结果就一板子题 思路:对每一个文件名开一棵线段树,然后树剖即可 #include<bits/stdc++.h> #define REP ...

  4. 【bzoj5084】hashit 广义后缀自动机+树链的并+STL-set

    题目描述 你有一个字符串S,一开始为空串,要求支持两种操作 在S后面加入字母C 删除S最后一个字母 问每次操作后S有多少个两两不同的连续子串 输入 一行一个字符串Q,表示对S的操作 如果第i个字母是小 ...

  5. Water Tree CodeForces 343D 树链剖分+线段树

    Water Tree CodeForces 343D 树链剖分+线段树 题意 给定一棵n个n-1条边的树,起初所有节点权值为0. 然后m个操作, 1 x:把x为根的子树的点的权值修改为1: 2 x:把 ...

  6. CF504E Misha and LCP on Tree(树链剖分+后缀树组)

    1A真舒服. 喜闻乐见的树链剖分+SA. 一个初步的想法就是用树链剖分,把两个字符串求出然后hash+二分求lcp...不存在的. 因为考虑到这个字符串是有序的,我们需要把每一条重链对应的字符串和这个 ...

  7. Codeforces Round #329 (Div. 2) D. Happy Tree Party LCA/树链剖分

    D. Happy Tree Party     Bogdan has a birthday today and mom gave him a tree consisting of n vertecie ...

  8. [POJ3237]Tree解题报告|树链剖分|边剖

    关于边剖 之前做的大多是点剖,其实转换到边剖非常简单. 我的做法是每个点的点权记录其到父亲节点的边的边权. 只要solve的时候不要把最上面的点记录在内就可以了. Tree Description Y ...

  9. Spoj Query on a tree SPOJ - QTREE(树链剖分+线段树)

    You are given a tree (an acyclic undirected connected graph) with N nodes, and edges numbered 1, 2, ...

随机推荐

  1. Python 数据结构理解分享

    摘要:分享学习Python数据结构的一些理解,主要包含序列(如列表和元组),映射(如字典)以及集合3中基本的数据结构,以及可变和不可变数据类型. Python 中的数据结构是根据某种方式将数据元素组合 ...

  2. windows下捕获本地回环网络中的报文RawCap

    一.下载地址: 官网地址:https://www.netresec.com/?page=RawCap 百度云:链接:https://pan.baidu.com/s/1mWCOTRF5XicuJitBA ...

  3. Java基础IO类之File类

    大三了,目前基础太差了,重新学习过!代码如下,里面都有详细的解释每一行代码代表的意思~ package IODemo; import java.io.File; import java.io.File ...

  4. IDEA 导入 NodeJS 项目部署启动

    1.导入项目 2.添加模块 3.配置启动项 4.启动 5.备注 如果不明白,新建一个项目查看配置详情 原文地址:https://blog.csdn.net/tiankongzhichenglyf/ar ...

  5. MySQL 索引机制

    MySQL 原理篇 MySQL 索引机制 MySQL 体系结构及存储引擎 MySQL 语句执行过程详解 MySQL 执行计划详解 MySQL InnoDB 缓冲池 MySQL InnoDB 事务 My ...

  6. iOS - Objective-C 关联(objc_setAssociatedObject、objc_getAssociatedObject、objc_removeAssociatedObjects)

    关联是指把两个对象相互关联起来,使得其中的一个对象作为另外一个对象的一部分. 关联特性只有在Mac OS X V10.6以及以后的版本上才是可用的. 在类的定义之外为类增加额外的存储空间 使用关联,我 ...

  7. javascript中bind()、call()、apply()的使用

    一直以来对bind().apply().call()这三个方法都模模糊糊的,现在有时间详细的看看这三个方法,并记录下来. bind() 参考文档:https://developer.mozilla.o ...

  8. ubuntu18.04 下启动Android Studio报错KVM is required to run this AVD. /dev/kvm device: permission denied.

    在ubuntu18.04下安装Android Studio,安装了模拟器后运行报错 KVM is required to run this AVD. /dev/kvm device: permissi ...

  9. Python语言程序设计:Lab5

    Programming Create a Class Student which should have the following information:Parameters passed to ...

  10. Mybatis.NET Oracle 线上神奇问题:Value does not fall within the expected range.

    1.错误现象 在向数据库查询一条数据的时候报如下错误: Value does not fall within the expected range. at Oracle.ManagedDataAcce ...