最近公共祖先的离线算法

离线算法即将所有询问一并读入,同时处理,同时回答,只要注意保存每个询问的时间顺序即可。

在线算法即对于每个询问及时处理,及时回答。

对于处理最近公共祖先的算法有很多种,tarjan算法,倍增算法,或者将lca转化成RMQ问题。 其中tarjan算法属于离线算法。

个人认为tarjan算法巧妙地利用了dfs的深度优先搜索的特性,对于一棵树的节点进行dfs遍历

我们用黑,白,灰三种颜色来表示节点的状态:

   白色: 未访问到该节点。

   灰色: 该节点已访问但已该节点为根的子树还没有完成访问 。

   黑色: 该节点以及以该节点为根的子树也已完成访问。

                              如图  当前访问到红色节点。

可见,所有的灰色节点都是当前访问点的祖先。而与当前点有关的最近公共祖先必然存在于这些灰色节点中,tarjan算法的具体算法步骤如下:

      <1> 遍历以某节点为根的子树。

      <2> 当该节点的儿子节点变为黑色时,将儿子节点所在的集合并入该节点所在的集合。

      <3> 当该节点变为黑色时,处理与该节点有关的询问。

      <4> 返回上一层。

设该节点为l , 询问的点对为l,r , 我们讨论一下r的颜色

若r此时的颜色为白色,则此询问当前还不能回答,

若r此时的颜色为灰色,依据之前所说,r一定是l的祖先,所以该询问的答案便是 r (此时r是r所在集合的root)

若r此时的颜色为黑色,若r不为l的子孙,r所在集合的root一定为一个灰色节点,而此灰色节点也一定是l的祖先,此节点便是最近公共祖先 ; 若r是l的子孙,该询问已经回答过,并且此时r所在集合的root是l , 答案依然正确。

附上tarjan算法的核心代码

   1:  void dfs(int i)
   2:  {
   3:      f[i]=i ; vis[i]=1;
   4:      for (int j=ad[i]; j; j=pre[j])
   5:      if (!vis[v[j]])
   6:      {
   7:        dfs(v[j]);
   8:        f[v[j]]= i ;
   9:      }
  10:      for (int j=head[i]; j ; j=link[j])
  11:      if (vis[next[j]])
  12:      {
  13:          ans[pos[j]] =find(next[j]);
  14:      }
  15:  }
:

转载于:https://www.cnblogs.com/ZGS1994/p/3546306.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Tarjan算法是一种用于求解最近公共祖(Least Common Ancestors,LCA)问题的离线算法算法的核心思想是利用深度优先搜索(DFS)和并查集(Union Find)来解决问题。 首先,我们从根节点开始遍历每一个节点,并将节点分为三类,用st[]数组表示。0代表还未被遍历,1代表正在遍历这个点,2代表已经遍历完这个点并且回溯回来了。这样的划分有助于确定节点的最近公共祖先。 在Tarjan算法中,我们一边遍历一边回应查询。每当遍历到一个节点时,我们查找与该节点相关的所有查询。如果查询中的节点已经被遍历完(即st[]值为2),我们可以利用已经计算好的信息来计算它们的最近公共祖先最近公共祖先的距离可以通过两个节点到根节点的距离之和减去最近公共祖先节点到根节点的距离来计算。 在Tarjan算法中,我们可以通过深度优先搜索来计算dist[]数组,该数组表示每个节点到根节点的距离。我们可以利用父节点到根节点的距离加上边的权值来计算每个节点到根节点的距离。 最后,我们可以通过并查集来操作st[]数组。当遍历完一个节点的所有子树后,将子树中的节点放入该节点所在的集合。这样,每个子树的节点的最近公共祖先都是该节点。 综上所述,Tarjan算法利用DFS和并查集来求解最近公共祖先问题。它的时间复杂度为O(n+m),其中n是节点数,m是查询次数。通过该算法,我们可以高效地解决最近公共祖先的问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [最近公共祖先之tarjan](https://blog.csdn.net/qq_63092029/article/details/127737575)[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^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [【模版】Tarjan离线算法最近公共祖先(LCA)](https://blog.csdn.net/weixin_43359312/article/details/100823178)[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^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [Tarjan算法求解最近公共祖先问题](https://blog.csdn.net/Yeluorag/article/details/48223375)[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^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值