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


树的结点定义

private static class TreeNode {
int val; List<TreeNode> children = new LinkedList<>(); public TreeNode() {
} public TreeNode(int val) {
this.val = val;
} @Override
public 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;
} @Override
public 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 10
public 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
// /
// 5
private 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
// /
// 5
private 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));
}
}

执行结果

【剑指Offer学习】【面试题50:树中两个结点的最低公共祖先】的更多相关文章

  1. (剑指Offer)面试题50:树中两个结点的最低公共祖先

    题目: 求树中两个结点的最低公共祖先 思路: 考虑一下几种情况: 1.该树为二叉搜索树 二叉搜索树是排序树,位于左子树点的结点都比父结点小,而位于右子树的结点都比父结点大,只需要从树的根结点开始和两个 ...

  2. 剑指Offer - 九度1509 - 树中两个结点的最低公共祖先

    剑指Offer - 九度1509 - 树中两个结点的最低公共祖先2014-02-07 01:04 题目描述: 给定一棵树,同时给出树中的两个结点,求它们的最低公共祖先. 输入: 输入可能包含多个测试样 ...

  3. 【剑指Offer面试编程题】题目1509:树中两个结点的最低公共祖先--九度OJ

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

  4. 《剑指offer》第六十八题(树中两个结点的最低公共祖先)

    // 面试题68:树中两个结点的最低公共祖先 // 题目:输入两个树结点,求它们的最低公共祖先. #include <iostream> #include "Tree.h&quo ...

  5. 树中两个结点的最低公共祖先--java

    题目:对于任意一个树,不仅仅限于二叉树,求树中两个结点的最低公共祖先结点. 解析:对于任意一棵树,显然并不局限于二叉树,也就是说树的非叶子结点可能存在多个子节点.所以,我们可以定义两个链表结构,存储这 ...

  6. 【Java】 剑指offer(68) 树中两个结点的最低公共祖先

      本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 输入两个树结点,求它们的最低公共祖先. 思路 该题首先要和面试 ...

  7. 【Offer】[68] 【树中两个结点的最低公共祖先】

    题目描述 思路分析 测试用例 Java代码 代码链接 题目描述 输入两个树结点,求它们的最低公共祖先. [牛客网刷题地址]无 思路分析 该题首先要确定是否为二叉树,还要确定是否为二叉搜索树,是否有父指 ...

  8. 《剑指offer》面试题68 - I. 二叉搜索树的最近公共祖先

    问题描述 给定一个二叉搜索树, 找到该树中两个指定节点的最近公共祖先. 百度百科中最近公共祖先的定义为:"对于有根树 T 的两个结点 p.q,最近公共祖先表示为一个结点 x,满足 x 是 p ...

  9. [jobdu]树中两个结点的最低公共祖先

    http://ac.jobdu.com/problem.php?pid=1509 此题最直观的方法是两次DFS,分别找到这两个节点的path,然后遍历path1和path2做比较,找到最后一个共同的元 ...

随机推荐

  1. Android 开发笔记___复选框__checkbox

    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout ...

  2. scrolling 优化 避免卡顿

    让我们来瞧瞧在滚动时到底发生了什么.在理解这个问题之前,我们先简要的介绍下浏览器是如何向屏幕绘制内容的.这一切都是从 DOM 树(本质上就是页面中的所有元素)开始的.浏览器先检查拥有了样式的 DOM, ...

  3. 浅谈如何使用swfupload工具与struts2无缝相接

    笔者在网上查找流行的上传组件,swfupload引入眼帘,受到JavaEye的一篇文章启发,历时三天,加以研究,现将心得奉上,献礼JavaEye. 由于笔者才疏学浅,经验匮乏,介绍不深入,仅供菜鸟参考 ...

  4. Android开发之漫漫长途 Ⅴ——Activity的显示之ViewRootImpl的PreMeasure、WindowLayout、EndMeasure、Layout、Draw

    该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽量按照先易后难的顺序进行编写该系列.该系列引用了<Android开发艺术探索>以及<深入理解And ...

  5. openpose模型在AI challenge人体骨骼关键点检测的表现

    因为之前正好看了CMU在CVPR2017上的论文<Realtime Multi-Person 2D Pose Estimation using Part Affinity Fields>, ...

  6. 【转】Entity Framework 5.0系列之自动生成Code First代码

    在前面的文章中我们提到Entity Framework的“Code First”模式也同样可以基于现有数据库进行开发.今天就让我们一起看一下使用Entity Framework Power Tools ...

  7. [转]如何监测谁用了SQL Server的Tempdb空间

    Tempdb 系统数据库是一个全局资源,供连接到 SQL Server 实例的所有用户使用.在现在的SQL Server里,其使用频率可能会超过用户的想象.如果Tempdb空间耗尽,许多操作将不能完成 ...

  8. Docker 三剑客之 Compose

    Compose 项目是 Docker 官方的开源项目,负责实现对 Docker 容器集群的快速编排,开源地址:https://github.com/docker/compose Compose 中的两 ...

  9. git命令提交项目

    相关的操作命令,总是忘记,故在此记录下: 此为linux下的命, windows的话,去掉sudo即可 1.进入项目代码根目录,执行: sudo git init 把这个目录变成git可以管理的仓库. ...

  10. javaweb-2-Tomcat初步学习与使用

    一.Tomcat服务器简介(此点网上官方有详尽的解释,故此不赘述,以学习使用为主) Apache Jakarta的开源项目 JSP/Servlet容器 二.Tomcat的目录结构 三.启动和停止Tom ...