题目:求树中两个结点的最低公共祖先,此树不是二叉树,并且没有指向父节点的指针。

树的结点定义
private static class TreeNode {int val;List<TreeNode> children = new LinkedList<>();public TreeNode() {}public TreeNode(int val) {this.val = val;}@Overridepublic String toString() {return  val + "";}
}
题目解析

 假设还是输入结点F和H .

  我们首先得到一条从根结点到树中某一结点的路径,这就要求在遍历的时候,有一个辅助内存来保存路径.比如我们用前序遍历的方法来得到从根结点到H 的路径的过程是这样的:( 1 )遍历到A,把A 存放到路径中去,路径中只有一个结点A; ( 2 )遍历到B,把B 存到路径中去,此时路径为A->B; ( 3 )遍历到D,把D 存放到路径中去,此,时路径为A->B->D; ( 4 ) :遍历到F,把F 存放到路径中去,此时路径为A->B->D->F;( 5) F 已经没有子结点了,因此这条路径不可能到这结点H. 把F 从路径中删除,变成A->B->D; ( 6 )遍历G. 和结点F 一样,这条路径也不能到达H. 边历完G 之后,路径仍然是A->B->D; ( 7 )由于D 的所有子结点都遍历过了,不可能到这结点H,因此D 不在从A 到H 的路径中,把D 从路径中删除,变成A->B; ( 8 )遥历E,把E 加入到路径中,此时路径变成A->B->E, ( 9 )遍历H,已经到达目标给点, A->B->E 就是从根结点开始到达H 必须经过的路径。
  同样,我们也可以得到从根结点开始到达F 必须经过的路径是A->B功。接着,我们求出这两个路径的最后公共结点,也就是B. B这个结点也是F 和H 的最低公共祖先.
  为了得到从根结点开始到输入的两个结点的两条路径,需要追历两次树,每边历一次的时间复杂度是O(n).得到的两条路径的长度在最差情况时是0(时,通常情况丁两条路径的长度是O(logn).

注意:可以在只遍历树一次就找到两个结点的路径,这部分留给读者自己去完成。

代码实现
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;public class Test50 {/*** 树的结点定义*/private static class TreeNode {int val;List<TreeNode> children = new LinkedList<>();public TreeNode() {}public TreeNode(int val) {this.val = val;}@Overridepublic String toString() {return val + "";}}/*** 找结点的路径** @param root   根结点* @param target 目标结点* @param path   从根结点到目标结点的路径*/public static void getNodePath(TreeNode root, TreeNode target, List<TreeNode> path) {if (root == null) {return;}// 添加当前结点path.add(root);List<TreeNode> children = root.children;// 处理子结点for (TreeNode node : children) {if (node == target) {path.add(node);return;} else {getNodePath(node, target, path);}}// 现场还原path.remove(path.size() - 1);}/*** 找两个路径中的最后一个共同的结点** @param p1 路径1* @param p2 路径2* @return 共同的结点,没有返回null*/public static TreeNode getLastCommonNode(List<TreeNode> p1, List<TreeNode> p2) {Iterator<TreeNode> ite1 = p1.iterator();Iterator<TreeNode> ite2 = p2.iterator();TreeNode last = null;while (ite1.hasNext() && ite2.hasNext()) {TreeNode tmp = ite1.next();if (tmp == ite2.next()) {last = tmp;}}return last;}/*** 找树中两个结点的最低公共祖先* @param root 树的根结点* @param p1 结点1* @param p2 结点2* @return 公共结点,没有返回null*/public static TreeNode getLastCommonParent(TreeNode root, TreeNode p1, TreeNode p2) {if (root == null || p1 == null || p2 == null) {return null;}List<TreeNode> path1 = new LinkedList<>();getNodePath(root, p1, path1);List<TreeNode> path2 = new LinkedList<>();getNodePath(root, p2, path2);return getLastCommonNode(path1, path2);}public static void main(String[] args) {test01();System.out.println("==========");test02();System.out.println("==========");test03();}// 形状普通的树//             1//           /   \//         2      3//        /         \//      4            5//     / \        /  |  \//    6   7      8   9  10public static void test01() {TreeNode n1 = new TreeNode(1);TreeNode n2 = new TreeNode(2);TreeNode n3 = new TreeNode(3);TreeNode n4 = new TreeNode(4);TreeNode n5 = new TreeNode(5);TreeNode n6 = new TreeNode(6);TreeNode n7 = new TreeNode(7);TreeNode n8 = new TreeNode(8);TreeNode n9 = new TreeNode(9);TreeNode n10 = new TreeNode(10);n1.children.add(n2);n1.children.add(n3);n2.children.add(n4);n4.children.add(n6);n4.children.add(n7);n3.children.add(n5);n5.children.add(n8);n5.children.add(n9);n5.children.add(n10);System.out.println(getLastCommonParent(n1, n6, n8));}// 树退化成一个链表//               1//              ///             2//            ///           3//          ///         4//        ///       5private static void test02() {TreeNode n1 = new TreeNode(1);TreeNode n2 = new TreeNode(2);TreeNode n3 = new TreeNode(3);TreeNode n4 = new TreeNode(4);TreeNode n5 = new TreeNode(5);n1.children.add(n2);n2.children.add(n3);n3.children.add(n4);n4.children.add(n5);System.out.println(getLastCommonParent(n1, n4, n5));}// 树退化成一个链表,一个结点不在树中//               1//              ///             2//            ///           3//          ///         4//        ///       5private static void test03() {TreeNode n1 = new TreeNode(1);TreeNode n2 = new TreeNode(2);TreeNode n3 = new TreeNode(3);TreeNode n4 = new TreeNode(4);TreeNode n5 = new TreeNode(5);TreeNode n6 = new TreeNode(6);n1.children.add(n2);n2.children.add(n3);n3.children.add(n4);n4.children.add(n5);System.out.println(getLastCommonParent(n1, n5, n6));}
}
运行结果

转载于:https://www.cnblogs.com/andy-zhou/p/6553626.html

50:树中两个结点的最低公共祖先相关推荐

  1. 树中两个结点的最低公共祖先

    题目描述: 给定一棵树,同时给出树中的两个结点,求它们的最低公共祖先. 输入: 输入可能包含多个测试样例. 对于每个测试案例,输入的第一行为一个数n(0<n<1000),代表测试样例的个数 ...

  2. 寻找树中两个结点的最低公共祖先

    寻找树中两个结点的最低公共祖先 Q:在树中寻找两个结点的最低公共祖先,是什么意思呢? A:树是由根节点衍生左右孩子继续衍生左右孩子的左右孩子.所以呢树中的两个结点是一定拥有最低公共祖先(这两个结点Q: ...

  3. Lowest Common Ancestor of a Binary Search Tree(树中两个结点的最低公共祖先)

    题目描述: Given a binary search tree (BST), find the lowest common ancestor (LCA) of two given nodes in ...

  4. 寻找二叉树中两个结点的最近公共祖先

    寻找二叉树中两个结点的最近公共祖先(这两个结点一定在树中) 对于二叉树中两个结点的最近公共祖先来说 ,这两个结点分别在自己的左子树和右子树中,所以要寻找二叉树中两个结点的最近公共祖先可以: 从根结点开 ...

  5. 求二叉树两个结点的最近公共祖先

    题目描述: 一颗二叉树按顺序存储结构进行存储,设计一个算法,求编号i和j的最近公共祖先 思路: 已知:二叉树.顺序存储 空的结点用"#"表示 如果i,j所在位置的结点数据不是&qu ...

  6. 二叉树两个结点的最低公共父结点 【微软面试100题 第七十五题】

    题目要求: 输入二叉树中的两个结点,输出这两个及诶单在数中最低的共同父结点. 题目分析: 还有一种情况:如果输入的两个结点中有一个或两个结点不在二叉树中,则输出没有共同父结点: 因此,可以在程序中定义 ...

  7. 二叉树找到两个结点的最近公共祖先

    面试题68 - I. 二叉搜索树的最近公共祖先 https://leetcode-cn.com/problems/er-cha-sou-suo-shu-de-zui-jin-gong-gong-zu- ...

  8. 二叉树中找两个结点的最近公共祖先结点

    一.搜索二叉树:第一变种是二叉树是一种特殊的二叉树:查找二叉树.也就是树是排序过的,位于左子树上的结点都比父结点小,而位于右子树的结点都比父结点大.我们只需要从根结点开始和两个结点进行比较.如果当前结 ...

  9. 树如何找共同祖先_如何找到任何二叉树中两个节点的最低公共祖先?

    小编典典 尼克·约翰逊是正确的,一个一个O(n)的时间复杂度算法是最好的,如果你没有父指针,你可以做.)对于算法的一个简单的递归版本中看到代码金丁的职务)它运行在O(n)的时间. 但是请记住,如果您的 ...

最新文章

  1. 完美,竟然用一个脚本就把系统升级到https了,且永久免费!
  2. 小甲鱼python课后作业十七_小甲鱼Python第十六讲课后习题--017函数
  3. 论文总结(negFIN: An efficient algorithm for fast mining frequent itemsets)
  4. 如何打造一个搞垮公司的产品?
  5. 基于Opencv--图像上采样
  6. LINUX下载编译libpq(postgresql)
  7. 龙腾P2P流媒体点播系统商业计划书
  8. github 设置语言为中文
  9. 市场28款主流同步整流DCDC芯片横向测评预告
  10. 利用python脚本将微信聊天信息提取到txt文件
  11. arcgis10之地理配准
  12. 深度学习平台配置 Pytorch+RTX3090+Pycharm
  13. python制作成绩分析系统_python实现学生成绩测评系统
  14. Linux内核发布时间表
  15. 中关村-DIY之国外网盘下载测试
  16. CC2530芯片介绍
  17. Java Language——IO 机制
  18. vue中nextTick使用引起的一个小问题
  19. ehviewer_ehviewer
  20. 【Adobe Acrobat】裁剪PDF文件中的一小部分并保存成单独页

热门文章

  1. 迷宫游戏c语言代码讲解,迷宫游戏C语言代码讲解.doc
  2. python3.8.2中文手册chm_springboot2.2.X手册:构建全局唯一的短链接数据中心
  3. 关于LR录制时不能自动启动IE浏览器的解决方法
  4. python识别人脸多种属性_OpenCV-Python(3)训练一个人脸识别器
  5. 恢复二叉搜索树Python解法
  6. 怎么把程序内部坐标转为屏幕坐标_全网最详细CNC加工中心程序代码大全!
  7. UE4学习-UE4结合vs2019混合编程
  8. python 批量下载图片_Python 批量下载图片示例
  9. html获取文本框中的文字,JavaScript实现input输入框点击获取文字内容
  10. 分布式锁的三种实现方式_分布式锁的几种实现方式~