学习二叉树 这一篇就够了 - java

学习二叉树 这一篇就够了 - java,第1张

学习二叉树 这一篇就够了 - java

文章目录

什么是树?

树形结构的概念重要概念 树的表示形式树的应用二叉树

概念

总结 两种特殊的二叉树二叉树性质二叉树存储模拟实现二叉树

准备工作

构造二叉树节点 - (孩子表示法)构建一棵这样的二叉树

debug【调试效果图】 二叉树最重要的功能 - 遍历二叉树 - (一级标题 更显重要。)

1. NLR:前序遍历(Preorder Traversal 亦称先序遍历)——访问根结点--->根的左子树--->根的右子树2. LNR:中序遍历(Inorder Traversal)——根的左子树--->根节点--->根的右子树3. LRN:后序遍历(Postorder Traversal)——根的左子树--->根的右子树--->根节点

小结练习程序实现 - 前中后序遍历

前序遍历拓展中序遍历后序遍历 [实战题 - LeetCode - 144. 二叉树的前序遍历 ](https://leetcode-cn.com/problems/binary-tree-preorder-traversal/) - 遍历思维

代码如下 [实战题 - LeetCode - 94. 二叉树的中序遍历](https://leetcode-cn.com/problems/binary-tree-inorder-traversal/) - 遍历思维

代码如下: [实战题 - LeetCode - 145. 二叉树的后序遍历](https://leetcode-cn.com/problems/binary-tree-postorder-traversal/)- 遍历思维拓展 : [实战题 - LeetCode - 144. 二叉树的前序遍历 ](https://leetcode-cn.com/problems/binary-tree-preorder-traversal/) - 子问题思路

代码如下

代码分析 4. 层序遍历练习题现在开始实现二叉树所具有的基本功能

获取二叉树的节点个数获取叶子结点个数获取第K层节点的个数 - 子问题思路获取二叉树的高度 - 子问题思路检测值为value的元素是否存在判断一棵树是不是完全二叉树拓展: 队列中 入队为元素为 null,队列也是有大小的。 与面试相关的二叉树OJ题型

[LeetCode - 100. 相同的树](https://leetcode-cn.com/problems/same-tree/)

题目解析解题思维代码如下 [LeetCode - 572. 另一棵树的子树](https://leetcode-cn.com/problems/subtree-of-another-tree/)

题目解析解题思维代码如下 [LeetCode - 104. 二叉树的最大深度](https://leetcode-cn.com/problems/maximum-depth-of-binary-tree//)[LeetCode - 110. 平衡二叉树](https://leetcode-cn.com/problems/balanced-binary-tree/)

题目解析代码如下进阶

代码如下 [LeetCode - 101. 对称二叉树](https://leetcode-cn.com/problems/symmetric-tree/)

题目解析解题思维代码如下: [牛客题霸 - KY11 二叉树遍历](https://www.nowcoder.com/practice/4b91205483694f449f94c179883c1fef?tpId=60&&tqId=29483&rp=1&ru=/activity/oj&qru=/ta/tsing-kaoyan/question-ranking)

题目解析解题思维解题步骤

程序框架creationTree 方法 【构造二叉树】 代码 如下 [LeetCode - 102. 二叉树的层序遍历](https://leetcode-cn.com/problems/binary-tree-level-order-traversal/)

解题思维

代码如下【不是题目的最终答案 - 过渡】 代码 如下 - 这是真货

代码 解析 [LeetCode - 236. 二叉树的最近公共祖先](https://leetcode-cn.com/problems/lowest-common-ancestor-of-a-binary-tree/)

题目解析解题思维

思维一:假设题目给定的是一棵二叉搜索树

代码如下 - 【二叉搜索树得出的某些结论,同样适用于本题】代码附图 思维二 - 假设 这个二叉树使用的是孩子双亲表示法来来表示

代码如下 [ 牛客题霸 - JZ36 二叉搜索树与双向链表](https://www.nowcoder.com/practice/947f6eb80d944a84850b0538bf0ec3a5?tpId=13&&tqId=11179&rp=1&ru=/activity/oj&qru=/ta/coding-interviews/question-ranking)

题目解析解题思维代码如下

附图 [LeetCode - 105. 从前序与中序遍历序列构造二叉树](https://leetcode-cn.com/problems/construct-binary-tree-from-preorder-and-inorder-traversal/)

解题思维

代码如下 [LeetCode - 106. 从中序与后序遍历序列构造二叉树](https://leetcode-cn.com/problems/construct-binary-tree-from-inorder-and-postorder-traversal/)

代码如下 [LeetCode - 606. 根据二叉树创建字符串](https://leetcode-cn.com/problems/construct-string-from-binary-tree/)

题目解析代码如下 拓展题

[ LeetCode - 144. 二叉树的前序遍历 ](https://leetcode-cn.com/problems/binary-tree-preorder-traversal/) - 非递归实现

解题思维 —— 借助栈代码如下 [ LeetCode - 94. 二叉树的中序遍历](https://leetcode-cn.com/problems/binary-tree-inorder-traversal/) - 非递归

代码如下 [LeetCode - 145. 二叉树的后序遍历](https://leetcode-cn.com/problems/binary-tree-postorder-traversal/)- 非递归

代码如下附图 本文结束

什么是树? 树形结构的概念

树是一种非线性的数据结构,它是由n(n>=0)个优先节点组成一个具有层次关系的集合。把它叫作树,是因为它看起来像一棵树,也就是说它是根朝上,而叶朝下。

它具有以下特点:
1、有一个特殊的结点,称为根结点,根节点没有前驱结点

2、除根结点外,其余结点被分成 M (M > 0) 个互不相交的集合 T1、 T2、…、 Tm,其中每一个集合 Ti(1 <= i <= m)又是一棵与树类似的子树。每棵子树的根结点有且只有一个前驱,可以有0个或多个后继。

3、 树是递归定义的。(这个等到 具体讲二叉树的时候会讲到的)
注意:树形结构中,子树之间不能有交集,否则就不是树形结构。


重要概念

结点的度:一个结点含有子树的个数;如上图:A的度为6。

树的度:一棵树中,所有结点度的最大值称为树的度;如上图:树的度为6。

叶子结点或终端结点:度为0的节点成为叶节点;如上图:B、C、H、I、P、Q、K、L、M、N都是叶子结点。



双亲结点或父结点:若一个结点含有子结点,则这个结点称为其子结点的父结点/双亲节点;如上图:A 是 B 的父结点。还有很多就不说了。



孩子结点或子结点:一个结点 含有的子树的根结点 称为 该节点 的 子结点;如上图: B 是 A 的孩子结点



根结点:一棵树中,没有双亲结点的结点;如上图:A

结点的层次:从根开始定义起,根为第一层,根的子结点为第2层,依次类推。


树的高度或深度:树中最大深度 ,即为高度;如上图:树的高度为 4,最大深度也为 4。但两者是不同的定西。
只有:当深度最大时, 深度才能说是高度。



非终端结点或分支结点: 度不为0的节点;如上图:D E F G J 都是分支结点。【简单来说:除了叶子结点【度为0的结点】,其它结点都是分支结点/非终端结点】

兄弟结点:具有相同父结点 的 节点 互称为兄弟结点;如上图:B C 互为兄弟结点。


堂兄弟结点:双亲/父 在同一层的结点互为堂兄弟; 如上图:H I 互为堂兄弟结点。


结点的祖先:从该节点出发,可以经过所有分支上的节点;如上图:A 是 所有节点的祖先。

子孙:以某个结点为根的子树中任一结点都称为该结点的子孙。如上图:所有结点都是A节点的子孙。

森林: 由 m(m >= 0)棵 互不相交的树组成的集合称为森林


树的表示形式

树结构相对线性表就比较复杂了,要存储表示起来就比较麻烦了,实际中树有很多种表示方式.
如:双亲表示法、孩子表示法、孩子双亲表示法、孩子兄弟表示法等等。
我们这里就简单的了解其中最常用的孩子兄弟表示法。

这里,我们在额外的讲一下 双亲表示法,就是 孩子兄弟法,加了一个存储空间,用来存储自身的父结点


树的应用

文件系统管理(目录 和 文件)

试想一下:我们的电脑常有空文件夹,点进去,就没有东西点了。对不对?那这空文件夹是不是就可以看作是叶子结点!【没有子结点,或者“度”为零嘛。】

再来举一个例子:


二叉树

概念

一棵二叉树是结点的一个有限集合,该集合:
1、可能为 空
2、可能是由一个根结点加上两棵别称为 左子树 和 右子树 的 二叉树组成。

二叉树 跟我们前面讲的树的区别就在于:二叉树 的 每个结点,最多只能有 两个 “孩子”/子树.。最少 零个。
也就是说:一棵树,如果是二叉树,那么它的每棵子树都是 二叉树【都有左子树 和 右子树】。


总结

1. 二叉树不存在度大于2 的结点
2.二叉树的子树有左右之分,次序不能颠倒,因此二叉树为有序树。
注意:对于任意的二叉树都是由以下几种情况复合而成的

上图的这四棵树都是二叉树


两种特殊的二叉树

1、满二叉树:一棵二叉树,如果每层的结点数都达到最大值,则这棵二叉树就是满二叉树。也就是说:如果一棵二叉树的层数为K,且结点总数是 2的k次方-1,则它就是满二叉树。


2、完全二叉树:完全二叉树是效率很高的数据结构,完全二叉树是由满二叉树引申出来的。对于深度为K的,有n个节点的二叉树,当其每一个结点都与深度为k的满二叉树中编号从 0 值 n -1 的结点 一一对应时称之为完全二叉树。值得注意的是 满二叉树 是一种特殊的 完全二叉树。
【简单来说: 给二叉树编号,根结点为0,然后下一层次 从左往右,依次编号。中间不能有间断,依次类推】


二叉树性质

性质1、 若规定根结点的层数为1,则一棵非空二叉树的第i层上,最多有 2的 (i - 1)次方。【这个我们在前面推导满二叉树的时候,已经出来了 2的(k-1)次方】
性质2、若规定只有根结点的二叉树的深度为1,则深度为k的二叉树的最大结点数是 2的K次方减一(k>=0)。【就是满二叉树】
性质3、对于任何一棵二叉树,如果其 叶结点个数 为 n0,度为2的非叶结点个数为 n2,则有 n0 = n2 + 1。【也就是说:叶结点的个数 总是比 非叶结点的个数多一个】


得出结论:任何一棵二叉树的叶子结点 比 度为2的节点 多一个。


来看三个练习题



性质4、具有n个结点的完全二叉树的深度 k 为 log以为2底的(n+1)上取整

再来看个练习题



性质5. 对于具有n个结点的完全二叉树,如果按照从上至下从左至右的顺序对所有节点从0开始编号,则对于序号为i
的结点有:
若 i > 0,双亲序号:(i-1)/2;i=0,i为根结点编号,无双亲结点
若2i+1 若2i+2


二叉树存储

二叉树的存储结构分为:顺序存储 和 类似于链表的链式存储。
这里,我们讲链式存储。
二叉树的链式存储是通过一个一个的节点引用起来的。常见的表示方式有二叉 和 三叉表示方式。
【二叉 : 孩子表示法;三叉 :孩子双亲表示法】


模拟实现二叉树

提前说明:二叉树的构建是一个非常复杂的过程,因为目前作者对二叉树的理解,还不是很深。所以,我们先会创建一个二叉树,但是这种创建方式,很LOW, 只是为了应付前期使用,比较简单,不是正确的常用创建方式。


准备工作 构造二叉树节点 - (孩子表示法)


构建一棵这样的二叉树


debug【调试效果图】


二叉树最重要的功能 - 遍历二叉树 - (一级标题 更显重要。)

学习二叉树结构,最简单的方式就是遍历。所谓遍历(Traversal)是指沿着某条搜索路线,依次对树中每个结
点均做一次且仅做一次访问。访问结点所做的 *** 作依赖于具体的应用问题【打印节点值,求节点个数等】。所有的二叉树相关的题目,基本上都是需要通过某种遍历去解题的。


1. NLR:前序遍历(Preorder Traversal 亦称先序遍历)——访问根结点—>根的左子树—>根的右子树


2. LNR:中序遍历(Inorder Traversal)——根的左子树—>根节点—>根的右子树

它的遍历顺序 就是 左子树 》》 根结点 》》 右子树


3. LRN:后序遍历(Postorder Traversal)——根的左子树—>根的右子树—>根节点

遍历顺序: 左子树 》》 右子树 》》 根结点


小结

前中后,无论是哪一种遍历,都是按照下面图的路线,来走的,只是打印的顺序不同而已,


练习

写出下面二叉树的 前中后排序的 序列

前序遍历:A B D E H C F G
中序遍历:D B H E A F C G
后序遍历:D H E B F G C A


程序实现 - 前中后序遍历

接着模拟实现二叉树,完成其遍历功能


前序遍历


拓展

二叉树的题 天生就是用递归来写的。【90%】
剩余 10%,使用非递归来解决问题。【只针对部分题】
其实所有的递归 都是 跟栈有关系的。


中序遍历

与前序同理,就是打印顺序不同.

    // 中序序遍历
    public void inorderTraversal(BTNode root){
        if(root == null){
            return;
        }
        inorderTraversal(root.left);
        System.out.print(root.val + " ");
        inorderTraversal(root.right);

    }


后序遍历

与前序同理,就是打印顺序不同.

    // 后序遍历
    public void postorderTraversal(BTNode root){
        if(root == null){
            return;
        }
        postorderTraversal(root.left);
        postorderTraversal(root.right);
        System.out.print(root.val + " ");
    }


实战题 - LeetCode - 144. 二叉树的前序遍历 - 遍历思维

这题,你可以直接将我们刚才的程序,复制粘贴。
唯一需要注意的就是

不过也很好解决。new一个 List 的嘛。反正它只用来存储前序遍历的结果序列。

其次,就是把输出语句 换成 add 添加 语句,将遍历的结果存入


代码如下
class Solution {
    List list;
    public List preorderTraversal(TreeNode root) {
        list =  new ArrayList<>();
        preorder(root);
        return list;
    }
    public void preorder(TreeNode root){
        if(root == null){
            return;
        }
        list.add(root.val);
        preorder(root.left);
        preorder(root.right);
    }
}


实战题 - LeetCode - 94. 二叉树的中序遍历 - 遍历思维

做法 完全一样,改变一下,add 添加数据的代码位置就行了

代码如下:
class Solution {
    List list;
    public List inorderTraversal(TreeNode root) {
        list = new ArrayList<>();
        inorder(root);
        return list;
    }
    public void inorder(TreeNode root){
        if(root == null){
            return;
        }
        inorder(root.left);
        list.add(root.val);
        inorder(root.right);
    }
}


实战题 - LeetCode - 145. 二叉树的后序遍历- 遍历思维

我就直接上代码。。。。

class Solution {
    List list;
    public List postorderTraversal(TreeNode root) {
        list = new ArrayList<>();
        postorder(root);
        return list;
    }
    public void postorder(TreeNode root){
        if(root == null){
            return;
        }
        postorder(root.left);
        postorder(root.right);
        list.add(root.val);
    }
}


拓展 : 实战题 - LeetCode - 144. 二叉树的前序遍历 - 子问题思路

将二叉树 整体分为 根结点,左子树 和 右子树 三部分,将其按照前序的遍历顺序进行添加。

代码如下
class Solution {
    public List preorderTraversal(TreeNode root) {
        List list = new ArrayList<>();
        if(root == null){
            return list;
        }
        list.add(root.val);
        List treeLeft = preorderTraversal(root.left);
        list.addAll(treeLeft);
        
        List treeRight = preorderTraversal(root.right);
        list.addAll(treeRight);
        return list;
    }
}


代码分析

4. 层序遍历

除了先序遍历、中序遍历、后序遍历外,还可以对二叉树进行层序遍历。设二叉树的根节点所在
层数为1,层序遍历就是从所在二叉树的根节点出发,首先访问第一层的树根节点,然后从左到右访问第2层
上的节点,接着是第三层的节点,以此类推,自上而下,自左至右逐层访问树的结点的过程就是层序遍历


练习题


现在开始实现二叉树所具有的基本功能 获取二叉树的节点个数


获取叶子结点个数


获取第K层节点的个数 - 子问题思路


获取二叉树的高度 - 子问题思路


检测值为value的元素是否存在

我们都知道在数组当中 查找一个数据:遍历数组,依次查找。
如果是链表中,是不是也是需要从头节点开始遍历比较,确认是不是存在的。
而现在是二叉树,是不是也需要遍历二叉树紫红的节点,确定某一个节点的值,是不是我们要找的。
最简单就是运用前序遍历来实现。先判断根结点值是否与 value值相等,其次 左子树,最后是右子树。

    // 检测值为value的元素是否存在
    public BTNode find(BTNode root,char value){
        if(root == null){
            return null;
        }
        if(root.val == value){// 判断根结点是不是 指定的数据
            return  root;
        }
        BTNode node1 = find(root.left,value);// 查找 左子树 是有存在 指定value值的节点【找到了,返回其引用,否,则返回null】
        BTNode node2 = find(root.right,value);// 查找 右子树 是有存在 指定value值的节点【找到了,返回其引用,否,则返回null】
        return node1 == null ? node2:node1;// 如果 node1 为 null,说明左子树没有指定value值的节点,返回 node2【右子树的查找的结果】
        // 如果 node2 也没有,也没有关系,反正都是返回null,node2找到了更好。返回其节点的引用
    }


判断一棵树是不是完全二叉树


拓展: 队列中 入队为元素为 null,队列也是有大小的。


与面试相关的二叉树OJ题型 LeetCode - 100. 相同的树


题目解析

注意我框选的题干部分【如果两个树在结构上相同,并且节点具有相同的值,则认为它们是相同的。】 它的意思: 两棵二叉树的节点位置 与 节点值 都必须一模一样。 两个条件中,只要有一个条件不符合。那么,这棵二叉树就不说是相同的树。


解题思维

利用遍历思想 去完成这道题:就是 使用同一种遍历方式,来遍历两棵二叉树节点,如果两棵树的节点值,有一组不相等就返回false。
但是!我们需要注意几个特殊情况!
情况1:
两棵二叉树为 空树【两棵二叉树的根结点为 null】

情况2:
一棵二叉树为null,另一棵不为null。


代码如下
class Solution {
    public boolean isSameTree(TreeNode p, TreeNode q) {
        if(p == null && q == null){// 两棵二叉树都为空树
            return true;
        } 
        // 两棵儿茶素中,有一颗树为空树
        if( (p != null && q == null) || (p == null && q != null) ){
            return false;
        }
        // 判断 两棵二叉树中,两个相对应的位置节点val值是否相同【宏观:判断根结点 是否相同】
        if(p.val != q.val){
            return false;
        }
        // 宏观:
        // 走到这里 说明两棵二叉树的头节点相同。接着,就是遍历 两棵二叉树的 左右子树。
        // 必须 两棵二叉树的左右子树都相等【true】。再结合根结点。【true】
        // 这两棵二叉树才能算是相同的树
        return isSameTree(p.left,q.left) && isSameTree(p.right,q.right); 
    }
}


LeetCode - 572. 另一棵树的子树


题目解析

简单来说: 在 根结点为 root 的 二叉树中,寻找有没有 一棵 与 根结点为subRoot的二叉树 相同的树。


解题思维

既然是寻找相同的树,那么,肯定是需要一个方法来判断 相同的树。【上一题的代码,就可以直接复制过来,作为一个判断相同树的方法来使用】

subRoot 为 root 的子树,无非就是以下几种情况:
1、root 与 subRoot 是一棵相同的树

2、 subRoot 是 root 的 左子树 ,或者说 是 左子树的一部分,

3、 subRoot 是 root 的 右子树 ,或者说 是 右子树的一部分,


注意:以上这些 *** 作,都可以交由我们的判断相同的方法来 *** 作。但这不意味着我们可以什么都不用做!


代码如下
class Solution {
    public boolean isSameTree(TreeNode p, TreeNode q) {
        if(p == null && q == null){// 两棵二叉树都为空树
            return true;
        } 
        // 两棵儿茶素中,有一颗树为空树
        if( (p != null && q == null) || (p == null && q != null) ){
            return false;
        }
        // 判断 两棵二叉树中,两个相对应的位置节点val值是否相同【宏观:判断根结点 是否相同】
        if(p.val != q.val){
            return false;
        }
        // 宏观:
        // 走到这里 说明两棵二叉树的头节点相同。接着,就是遍历 两棵二叉树的 左右子树。
        // 必须 两棵二叉树的左右子树都相等【true】。再结合根结点。【true】
        // 这两棵二叉树才能算是相同的树
        return isSameTree(p.left,q.left) && isSameTree(p.right,q.right); 
    }

    public boolean isSubtree(TreeNode root, TreeNode subRoot) {
        if(root == null){
            return subRoot!=null ? false:true;
        }
        // 判断 是否是同一棵树
        if(isSameTree(root,subRoot)){
            return true;
        }
        // 判断 subRoot是否 是 root的左子树 或者 是左子树的一部分
        if(isSubtree(root.left,subRoot)){
            return true;
        }
        // 判断 subRoot是否 是 root的右子树 或者 是右子树的一部分
        if(isSubtree(root.right,subRoot)){
            return true;
        }
        return false;
    }

}


LeetCode - 104. 二叉树的最大深度

这题就是我们前面实现的 求二叉树高度功能。【高度 等价于 二叉树的最大深度】
这里我就直接上代码了。

class Solution {
    public int maxDepth(TreeNode root) {
        if(root == null){
            return 0;
        }
        return  Math.max(maxDepth(root.left),maxDepth(root.right)) + 1;
    }
}


LeetCode - 110. 平衡二叉树


题目解析

题目的意思: 让我们去计算 这棵二叉树的左右子树的高度。其高度差不超过 1。
唯一需要注意的是:平衡二叉树 不仅仅是 左右子树 是这么个情况,还需要把左右子树 看作一个 二叉树,而且也必须是一个平衡二叉树。
也就是说:平衡二叉树 的 左右子树高度差 不超过 1,且如果将左右子树看作 是 一棵单独二叉树 ,也是一棵平衡二叉树。


代码如下
 时间复杂度 O(N)
 空间复杂度 O(log2N)
class Solution {
    public boolean isBalanced(TreeNode root) {
        if(root == null){
            return true;
        }
        int leftHeight = getHeight(root.left);// 获取 左子树高度
        int rightHeight = getHeight(root.right);// 获取 右子树高度
        // 判断 左右子树的高度差不能超过 1。另外,左右子树自身  也要满足这个条件。
        return Math.abs(leftHeight - rightHeight) <= 1 && isBalanced(root.left) && isBalanced(root.right);  

    }
    public int getHeight(TreeNode root){// 计算根结点为“root”的二叉树高度
        if(root == null){
            return 0;
        }
        return Math.max(getHeight(root.left),getHeight(root.right)) + 1;
    }
}


进阶


代码如下
class Solution {
    public boolean isBalanced(TreeNode root) {
        if(root == null){
            return true;
        }
        return getHeight(root)> -1;
    }
    public int getHeight(TreeNode root){
        if(root == null){
            return 0;
        }
        int L = getHeight(root.left);// 左子树高度
        int R = getHeight(root.right);// 右子树高度
        if(L>=0 && R>=0 && Math.abs(L-R)<=1){
            return Math.max(L,R) + 1;
        }else{// 说明不平衡
            return -1;
        }
    }
}


LeetCode - 101. 对称二叉树


题目解析

简单来说:以根结点为分界线,将二叉树的左右子树分割开来,并以此线 作为对称轴。
对称简单来说:将这上图看作一张纸,沿着 对称轴 折叠,你会发现 左右子树的节点完美重合【位置和值都是】


解题思维

简单来说:以二叉树的根结点 1 为 分割线, 去判断 左右子树 节点的值是否对称。
对称图如下:

但是有几个特殊情况需要注意!
1、二叉树为空树,根结点为null。【空树也是树,也算对称树】

2、二叉树只有一个根结点,也算对称二叉树【左右两个空子树】

3、 二叉树的左右子树中:有一个为空子树【非对称子树】


代码如下:
class Solution {
    public boolean isSymmetric(TreeNode root) {
        // 二叉树为 空树
        if(root == null){
            return true;
        }
        return isSymmetricChild(root.left,root.right);
    }
    public boolean isSymmetricChild(TreeNode L,TreeNode R){
        // 树的根结点 左右子树为null
        if(L == null && R == null){
            return true;
        }
        // 二叉树的左右子树有一棵子树为空树
        if( (L == null && R != null) || (L != null && R == null) ){
            return false;
        }
        // 继续判断:两棵树中 对称节点的值【左对右,右对左】
        if(L.val == R.val){
            return isSymmetricChild(L.left,R.right) &&isSymmetricChild(L.right,R.left);
        }
        return false;
    }
}


牛客题霸 - KY11 二叉树遍历


题目解析

简单来说:根据所给出 二叉树 前序遍历序列,构造出一棵二叉树。对其进行 中序遍历,输出结果。


解题思维

首先,我们先根据 它所给的 前序遍历序列 和 关键条件【# 代表 空树】 ,根据示例的 数据,来 推导构造出一棵二叉树

难点:是推导出来了,也画出来了,但是代码具体如何实现?
答案:既然 题目 给的是前序遍历的结果,我们也需要用前序遍历的方式来构建。


解题步骤 程序框架


creationTree 方法 【构造二叉树】

思路: 创建一个静态变量 i ,用来遍历字符串,辅助我们去构造 一个 二叉树。
不要抱有太大的异或,你就跟着我的思路一步步来。稳得很!

因为我们不知道这个字符串具体输入什么样的数据。所以重点就在于判断!
判断 我们拿到的字符串数据。


代码 如下
import java.util.*;
// 树节点
class TreeNode{
    public char val;
    public TreeNode left;
    public TreeNode right;
    public TreeNode(char val){
        this.val = val;
    }
}
    
public class Main{
    public static void main(String[] args){
        // 循环输入
        Scanner sc = new Scanner(System.in);
        while(sc.hasNextLine()){
            String  str = sc.nextLine();//读取 字符串
            TreeNode root = creationTree(str);// 构造二叉树返回其根结点
            inorderTraversal(root);// 中序遍历打印
        }
        sc.close();
    }
    // 构造二叉树
    public static int i = 0;
    public static TreeNode creationTree(String str){
       TreeNode  root = null;// 先定义 一个为null
        char ch = str.charAt(i);// 获取下标为 i  的 字符 数据
        if( ch != '#'){// ch 为字符数据时
            root  =  new  TreeNode(ch);// root 根据 ch 去创建一个节点
            i++;// 为下一次获取 字符 数据,做准备。
            root.left = creationTree(str);// 宏观:将 左子树接回根结点
            root.right = creationTree(str);// 宏观:将 右子树接回根结点
        }else{// ch  为 ‘#’ 字符时,代表空树【也就是null 节点】
            i++;// 不用管,直接加加。
        }
        // 如果 ch == ‘#’,刚好 我们 root 就是 null
        // 如果 ch  是 字符数据,不用担心,if语句帮我们处理好了。
        return root;
    }
    // 中序遍历【这个简单,我就先写了】
    public static void inorderTraversal(TreeNode root){
        if(root == null){
            return;
        }
        inorderTraversal(root.left);
        System.out.print(root.val + " ");
        inorderTraversal(root.right);
    }
}


 ;

LeetCode - 102. 二叉树的层序遍历


解题思维

这里我们需要 了解/实现 一下 层序遍历的代码。
思维: 跟前面判断 完全二叉树的方法一致:用队列实现。出队的时候,随便打印一下,不同的是 如果左右子树为空树,就不用入队了。【没有值可以打印】

代码如下【不是题目的最终答案 - 过渡】


代码 如下 - 这是真货

因为题目很gou!要求的 返回值,很那啥。
把 每一层的节点数据,先存入一个 线性表 中。
然后,根据层次顺序,存入 另一个线性表中。【你可以理解为 是 一个 二维数组】

但是,有一个问题:我们前面写的 层序遍历,是没有分层的。
只要出队的元素不为null,我就打印一下。
来下面的代码,看看我是怎么处理的

class Solution {
    public List> levelOrder(TreeNode root) {
        List> result = new ArrayList<>();
        if(root == null){
            return result;
        }
        Queue q = new linkedList<>();
        q.offer(root);
        while(!q.isEmpty()){
            int size = q.size();// 记录当前队列的元素个数
            List list = new ArrayList<>();// 存储当前层次的元素
            while(size > 0){// 将当前层次的元素 出队
                TreeNode tmp = q.poll();// 出队
                list.add(tmp.val);//  存入 list 中
                size--;// 表示 将当前层次的元素存入完成。
                // 同时,将该层次节点 的 下一层次的节点(左右子树) 存入
                if(tmp.left != null){
                    q.offer(tmp.left);
                } 
                if(tmp.right != null){
                    q.offer(tmp.right);
                }
            }
            result.add(list);// 将每层的 list 存入 result 中
        }
        // 返回 结果
        return result;
    }
}


代码 解析


 ;

LeetCode - 236. 二叉树的最近公共祖先


 ;

题目解析

给我们一棵二叉树,和 这棵二叉树中的两个节点,要求我们找到 这两个节点的 公共祖先【父亲节点,注意父亲节点也可以是 二个节点中的一个。也就是说另一个节点是孩子结点】


 ;

解题思维

 ;

思维一:假设题目给定的是一棵二叉搜索树


 ;

代码如下 - 【二叉搜索树得出的某些结论,同样适用于本题】
class Solution {
    public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
        // 终止条件
        if(root == null){
            return root;
        }
        // 根结点为 p 和 q 中一个
        if(root == q || root == p){
            return root == q ? q : p;
        }
        // 在 左子树中 寻找 p 和 q 中的一员 或者 公共祖先【可能没有找到:null】
        // 宏观:在左子树中 p 和 q 的公共祖先
        TreeNode leftNode = lowestCommonAncestor(root.left,p,q);
        // 在 右子树中 寻找 p 和 q中的一员 或者 公共祖先【可能没有找到:null】
          // 宏观:在右子树中 p 和 q 的公共祖先
        TreeNode rightNode = lowestCommonAncestor(root.right,p,q);
        // 如果在 root 的 左右子树寻找结果都不为null,说明 p 和 q 处于不同的子树当中
        if(leftNode != null && rightNode != null){
            return root;// 那么此时的根结点就是  公共祖先
        }
         // 如果在 root 的 左右子树寻找结果,有一个结果为null,说明 p 和 q 处于相同同的子树当中
         // 那么就有可能出现 p 是 q 的 公共祖先,或者 q 是 p 的公共祖先。
         // 在根据递归的性质:每个节点都扮演过根结点的角色
         // 所以,在 p 或 q 的公共祖先的情况下:root 递归到 到这个位置,就会直接返回 root.
         // 再从宏观出发: p 和 q 还有 它们的公共祖先,在一棵子树下
         // 很显然:另一棵子树 是不可能找到公共祖先节点的,所以 root 会递归到 null,并将其作为结果返回。
         // 我们只需去判断一棵子树的结果就够了。
        return  leftNode == null ? rightNode : leftNode;
    }
}


 ;

代码附图

如果根结点不为公共节点,就去 左右子树中 去找。

如果 根结点 就是 p 和 q 的一员,直接返回根结点。

需要注意的是:有可能 p 和 q 都在 左子树中;或者,都在右子树中。
且,可能存在 p 是 q 的 祖先节点的情况,或者, q 是 p 的 祖先节点情况。;


思维二 - 假设 这个二叉树使用的是孩子双亲表示法来来表示


代码如下
class Solution {
    public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
        if(root == null){
            return root;
        }
        Stack stackq = new Stack<>();
        Stack stackp = new Stack<>();
        // 获取路径
        getPath(root,p,stackp);
        getPath(root,q,stackq);
        // 获取栈的大小
        int sizep = stackp.size();
        int sizeq = stackq.size();

        // 利用快慢指针的思想,让栈大的一方先出栈
        // 直到两个栈的大小一致。然后,一起出栈
        // 如果 出栈的元素 相等,此元素就是 最近祖先节点
        int difference = sizep - sizeq;
        if(difference < 0 ){
            difference = sizeq - sizep;
            while(difference>0){
                stackq.pop();
                difference--;
            }
            while(!stackq.isEmpty() && !stackp.isEmpty() ){
                TreeNode nodeq = stackq.pop();
                TreeNode nodep = stackp.pop();
                if(nodeq == nodep){
                    return nodep;
                }
            }
        }else{
            while(difference>0){
                stackp.pop();
                difference--;
            }
            while(!stackq.isEmpty() && !stackp.isEmpty() ){
                TreeNode nodeq = stackq.pop();
                TreeNode nodep = stackp.pop();
                if(nodeq == nodep){
                    return nodep;
                }
            }
        }
        // 表示 这个二叉树 不存在 祖先节点
        return null;
    }
    // 获取 指定节点 的 存储路径
    public boolean getPath(TreeNode root,TreeNode node,Stack stack){
        if(root == null || node == null){
            return false;
        }
        stack.push(root);
        if(root == node){
            return true;
        }
        boolean flag =getPath(root.left,node,stack);
        if(flag){
            return true;
        }

        flag =getPath(root.right,node,stack);
        if(flag){
            return true;
        }
        stack.pop();
        return false;
    }
}


牛客题霸 - JZ36 二叉搜索树与双向链表


题目解析

将一颗搜索二叉树,进行有序排序【升序】。按照升序的顺序,创建一个链表【双向】。


解题思维

首先通过上面那题思维一,我们得知 搜索二叉树,如果对其进行中序遍历,其结果就是有序的。

最后一个问题: 将其结果转换成一个双向链表 【难点】


代码如下
public class Solution {
    public TreeNode Convert(TreeNode pRootOfTree) {
        if(pRootOfTree == null){
            return null;
        }
        // 获取 头节点位置 - 详尽见附图
        inorderTraversal(pRootOfTree);
        TreeNode head = pRootOfTree;
        while(head.left != null){
            head = head.left;
        }
        return head;
    }
    public TreeNode prev ;
    public void inorderTraversal(TreeNode root){
        if(root == null){
            return;
        }
        inorderTraversal(root.left);
        // 打印
        root.left = prev;
        if(prev != null){
            prev.right = root;
        }
        prev = root;
//         System.out.print(root.val + " ");
        inorderTraversal(root.right);
    }
}


附图


LeetCode - 105. 从前序与中序遍历序列构造二叉树

这题的意思,就不用我讲了。跟前面的那道选择题一样【根据 两种遍历方法(前中 或 中后)确定 一棵唯一的二叉树】,不同的是 选择要求返回 另外一种遍历的结果,而这题要我们返回 这棵二叉树的根结点。相同的是 根据两种遍历方式 确定 / 构造 唯一 的 二叉树。


解题思维

根据上面的附图,从而引申出我的解题逻辑。


代码如下
class Solution {
    public TreeNode buildTree(int[] preorder, int[] inorder) {
        // 根据 一种遍历结果是无法确定一棵唯一的二叉树的。
        if(preorder == null ||  inorder == null){
            return null;
        }
        return creationTree(preorder,inorder,0,inorder.length - 1);
    }
    public int rootIndex; // 遍历前序数组的变量,默认值为 0
    public TreeNode creationTree(int[] preorder,int[] inorder,int begin,int end){
        if(begin > end){
            return null;
        }
        TreeNode root = new TreeNode(preorder[rootIndex]);
        int inorderIndex = findIndex(inorder,begin,end,root.val);
        if(inorderIndex == -1){
            return null;
        }
        rootIndex++;
        root.left = creationTree(preorder,inorder,begin,inorderIndex-1);
        root.right = creationTree(preorder,inorder,inorderIndex+1,end);
        return root;  
    }
    private int findIndex(int[] inorder,int begin, int end,int key){
        for(int i = begin;i <= end;i++){
            if(key == inorder[i]){
                return i;
            }
        }
        return -1;
    }
}


LeetCode - 106. 从中序与后序遍历序列构造二叉树

这个就很简单了,就是上面那题改一些地方就行了。
preorder 换成 postorder,rootIndex 要赋值 【postorder.length - 1】
因为 后序遍历: 左 》 右 》根,也就是说:倒数第一个元素 是 根结点,倒数第二个 是 右子树的根结点。【也就是说,先创建 根结点,其次是 右子树,最后左子树】


代码如下
class Solution {
    public int rootIndex; // 遍历前序数组的变量,默认值为 0
    public TreeNode buildTree(int[] inorder,int[] postorder) {
        // 根据 一种遍历结果是无法确定一棵唯一的二叉树的。
        if(postorder == null ||  inorder == null){
            return null;
        }

        rootIndex = postorder.length - 1 ;
        
        return creationTree(postorder,inorder,0,inorder.length - 1);
    }
    public TreeNode creationTree(int[] postorder,int[] inorder,int begin,int end){
        if(begin > end){
            return null;
        }
        TreeNode root = new TreeNode(postorder[rootIndex]);
        int inorderIndex = findIndex(inorder,begin,end,root.val);
        if(inorderIndex == -1){
            return null;
        }
        rootIndex--;
        root.right = creationTree(postorder,inorder,inorderIndex+1,end);
        root.left = creationTree(postorder,inorder,begin,inorderIndex-1);
        return root;  
    }
    private int findIndex(int[] inorder,int begin, int end,int key){
        for(int i = begin;i <= end;i++){
            if(key == inorder[i]){
                return i;
            }
        }
        return -1;
    }
}


LeetCode - 606. 根据二叉树创建字符串


题目解析

题目大概的意思:将一个二叉树转换成一个由括号和整数组成的字符串。


代码如下
class Solution {
    public String tree2str(TreeNode root) {
        if(root == null){
            return "()";
        }
        StringBuilder sb = new StringBuilder();
        preOrderChange(root,sb);
        return sb.toString(); 
    }
    public void preOrderChange( TreeNode root, StringBuilder sb ){
        if(root == null){
            return;
        }
        sb.append(root.val);
        if(root.left != null){
            sb.append("(");
            preOrderChange(root.left,sb);
            sb.append(")");
        }else{
            // root.left == null;
            if(root.right == null){
                return ;
            }else{
                sb.append("()");

            }
        }
        if( root.right == null){
            return;
        }else{
            sb.append("(");
            preOrderChange(root.right,sb);
            sb.append(")");
        }
    }

}


拓展题

LeetCode - 144. 二叉树的前序遍历 - 非递归实现


解题思维 —— 借助栈


代码如下
// 非递归
class Solution {
    public List preorderTraversal(TreeNode root) { 
        List list = new ArrayList();
        Stack stack = new Stack<>();
        TreeNode cur = root;
        while(cur != null || !stack.isEmpty()){
            while(cur!= null){
                list.add(cur.val);
                stack.push(cur);
                cur = cur.left;
            }
            TreeNode tmp = stack.pop();
            cur = tmp.right;
        }
        return list;
    }
}


LeetCode - 94. 二叉树的中序遍历 - 非递归

有了前序之见。那么,中序将不存在问题。无非就是 list.add 语句换个位置。


代码如下
// 非递归
class Solution {
    public List inorderTraversal(TreeNode root) {
         List list = new ArrayList<>();
         Stack stack = new Stack<>();
         TreeNode cur = root;
         while(cur != null || !stack.isEmpty() ){
             while(cur!=null){
                 stack.push(cur);
                 cur = cur.left;
             }
             TreeNode tmp = stack.pop();
             list.add(tmp.val);
             cur = tmp.right;
         }
         return list;
    }
}


LeetCode - 145. 二叉树的后序遍历- 非递归


代码如下
// 非递归
class Solution {
    public List postorderTraversal(TreeNode root) {
        List list = new ArrayList<>();
        Stack stack = new Stack<>();
        TreeNode cur = root;
        TreeNode prev = null;// 见附图
        while(cur != null || !stack.isEmpty()){
            while(cur!=null){
                stack.push(cur);
                cur = cur.left;
            }
            TreeNode tmp = stack.peek();
            if(tmp.right == null || prev == tmp.right){// 见附图
                stack.pop();
                list.add(tmp.val);
                prev = tmp;
            }else{
                cur = tmp.right;
            }  
        }
        return list;
    }
}


附图


本文结束

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/5715466.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-12-17
下一篇 2022-12-17

发表评论

登录后才能评论

评论列表(0条)

保存