当前位置: 首页 > article >正文

【LeetCode】二叉树的中序遍历(递归,迭代,Morris遍历)

目录

题目要求:给定一个二叉树的根节点 root ,返回 它的 中序 遍历 。

方法一:递归

方法二:迭代

思路分析:

复杂度分析

代码展示:

方法三:Morris 遍历

思路分析:

复杂度分析

代码展示:


题目要求:给定一个二叉树的根节点 root ,返回 它的 中序 遍历 。

示例 1:

输入:root = [1,null,2,3]
输出:[1,3,2]

提示:

  • 树中节点数目在范围 [0, 100] 内
  • -100 <= Node.val <= 100

 

方法一:递归

递归的方法二叉树的前序遍历在之前的博客中已经写过,需要的小伙伴可以点击链接查看

递归求二叉树的前中后序遍历

【LeetCode】二叉树的前序遍历(递归,迭代,Morris遍历)

这篇文章主要来讲解非递归的方法对二叉树进行中序遍历

方法二:迭代

思路分析:


迭代的方式其实与递归是等价的,区别在于递归的时候隐式地维护了一个栈,而我们在迭代的时候

需要显式地将这个栈模拟出来,其余的实现与细节都相同,具体可以参考下面的代码

复杂度分析

  • 时间复杂度:O(n),其中 n 是二叉树的节点数。每一个节点恰好被遍历一次。

  • 空间复杂度:O(n),为迭代过程中显式栈的开销,平均情况下为 O(log⁡n),最坏情况下树呈现链状,为 O(n)

代码展示:

 public List<Integer> inorderTraversal(TreeNode root) {
        List <Integer> list = new ArrayList<>();
        Stack <TreeNode> stack = new Stack<>();

        //栈非空或者root非空
        while(root != null || !stack.isEmpty()){
            //先根后左入栈
            while(root != null){
                stack.push(root);
                root = root.left;
            }
            //此时root为空,说明上一个入栈的root没有左子树
            //没有左子树,可以出栈
            root = stack.pop();
            list.add(root.val);
            //此时判断右子树
            root = root.right;
        }
        return list;
    }

方法三:Morris 遍历

Morris 遍历使用二叉树节点中大量指向 null 的指针,由 Joseph Morris 于 1979 年发明。

思路分析:

将当前根节点的左侧最右侧节点的right指向当前根节点,省去了栈的维护,连接之后可以直接顺着节点遍历完整个二叉树,以下图为例:

6f9c88bc38954a898dd01d66029bf510.jpeg

复杂度分析

  • 时间复杂度:O(n),其中 n 是二叉树的节点数。每一个节点恰好被遍历一次。

  • 空间复杂度:O(1)

代码展示:

    public List<Integer> inorderTraversal(TreeNode root) {
        List<Integer> list = new ArrayList<>();
        if(root == null){
            return list;
        }
        TreeNode cur1 = root;
        TreeNode cur2 = null;
        while(cur1 != null){
            cur2 = cur1.left;
            if(cur2 != null){
                while(cur2.right != null && cur2.right != cur1){
                    cur2 = cur2.right;
                }
                //此时说明cur2走向了最右侧子树
                //1.还未连接,建立连接
                if(cur2.right != cur1){
                    cur2.right = cur1;
                    cur1 = cur1.left;
                    continue;
                //否则说明已经走过,断开连接
                }else{
                    cur2.right = null;
                    list.add(cur1.val);
                }
            }else{
                list.add(cur1.val);
            }

            cur1 = cur1.right;
        }
        return list;
    }


http://www.kler.cn/a/6068.html

相关文章:

  • 【Arm】Arm 处理器的半主机(semihosting)机制
  • 晨辉面试抽签和评分管理系统之一:考生信息管理和编排
  • PHP语言的数据库编程
  • HackMyVM-Again靶机的测试报告
  • 关于考完两门专业课的感受阶段性总结
  • 后台管理系统动态面包屑Breadcrumb组件的实现
  • SELECT COUNT(*) 会造成全表扫描吗
  • Disjoint 集合数据结构或 Union-Find 算法简介
  • jsp823科研项目申报管理网站cc94程序mysql+java
  • Uni-Mol: A Universal 3D Molecular Representation Learning Framework
  • 使用new bing chat成功了
  • 华为OD机试用java实现 -【数字的排列 or 数字反转打印】
  • CRM客户管理系统不被销售接受的五大原因
  • 二、MySQL 基础
  • 【软考——系统架构师】系统开发基础知识
  • 如何保证RocketMQ顺序消息以及可能出现的问题
  • Databend 开源周报第 86 期
  • 【CSS】清除浮动 ① ( 清除浮动简介 | 清除浮动语法 | 清除浮动 - 额外标签法 )
  • 计算机组成原理:5. 输入输出系统
  • Higress 0.7.0 版本发布:GA 进入倒计时
  • 学会吊打面试官之list
  • 通过两道一年级数学题反思自己
  • LeetCode222. 完全二叉树的节点个数(二分查找+二进制表示路径法)
  • 免 交 互
  • 2023年6月18日的CDGA/CDGP数据治理认证考试报名开始啦!
  • 主机系统扫描程序设计