赞
踩
LCA(Least Common Ancestors),即最近公共祖先,是指在有根树中,找出某两个结点x和y最近的公共祖先。
用三种算法可以求解LCA问题,分别为朴素算法、倍增算法和Tarjan算法。
倍增算法和Tarjan算法都在建立在朴素算法的思想下,因此,了解朴素算法的思想有助于更好的理解进阶算法。
朴素算法前置知识:邻接表,dfs。
假设我们要寻找某两个节点x和y的LCA,那么我们肯定是让深度更深的那个结点跳到另一个结点深度处,然后再让这两个结点一起向上跳,直到首次相遇。
光说可以有些抽象,举个例子,就以下面这张图中的树为例。
图中结点4先跳到结点3的位置,然后两个结点一起向上跳,随后跳到结点1处相遇,所以结点2和结点4的LCA为结点1。
也就是说,我们只要记录下每个结点的深度信息和祖先信息,就能通过逐个向上跳跃直至相遇来确定两个结点的LCA。
这便是LCA朴素算法的核心支撑所在,但是由于朴素算法每次跳跃一层,因此他的时间复杂度很差,尤其是当树退化为链的时候,那么如果我们让他跳跃多层,是不是可以更好更快的解决问题呢?答案是一定的,而接下来的倍增算法就是这个思想。
倍增算法前置知识:邻接表,DP&倍增,dfs。
倍增算法我们将会定义一个数组fa[i][j]表示结点i向上跳2j层所到的结点,从而实现了倍增跳跃,而且,通过有限的组合跳跃,我们到达任意结点处。例如向上5层,我们可以先向上跳跃22层之后,再向上跳跃2^0次方。同时,我们可以证明又fa[x][i] = fa[fa[x][i-1]][i-1],实质就是2^(i-1) + 2^(i-1) = 2^i.
同时,倍增LCA算法中还用到了贪心的思想,假如现在有两个结点x,y,假设x深度更大,则我们要尽可能地让x在不超过y的深度的前提下,尽可能地接近y,也就是跨的步子尽可能大!这样操作过后,结点x与结点y的深度就一定相同了。
相同之后,如果已经重合,直接return,如果没有,那么现在两个结点处于一个平行的位置。接下来我们让两个结点同时向上跳,也是能跳多大就跳多大,但是肯定是有限制的,就像上一步一样,这个限制就是只有在跳完之后他们结点不重合时才跳。这个地方有点绕,不要急,我们结合代码看一下。
int lca(int u, int v) { if(depth[u] < depth[v]) swap(u, v); for(int i = MAXLOG - 1; i >= 0; i--) { if(depth[u] - (1 << i) >= depth[v]) { u = parent[u][i]; } } if(u == v) return u; // 注意看这里,这一步是平行之后的操作 for(int i = MAXLOG - 1; i >= 0; i--) { if(parent[u][i] != parent[v][i]) { u = parent[u][i]; v = parent[v][i]; } } return parent[u][0]; }
Tarjan算法前置知识:邻接表,并查集,dfs。
Tarjan巧妙地将并查集和dfs结合在一起,实现了将单次查询地时间复杂度降到了常数级别的离线算法!
具体步骤如下:
离线用Tarjan,在线用倍增。:)
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。