二叉树的前中后序遍历(递归与迭代)

news2024/11/28 22:31:07

作者~小明学编程

文章专栏Java数据结构

格言目之所及皆为回忆,心之所想皆为过往

目录

简介

前序遍历

递归法

迭代法

中序遍历

递归法

迭代法

后序遍历

递归法

迭代法


简介

    前面学习二叉树的时候,已经学过的二叉树的前中后序的遍历,但是当时实现功能的时候采用的都是递归法,今天把迭代法也说一下,虽然二者的时间复杂度和空间复杂度一样,但是递归可能会导致栈爆了的风险,当然对于我来说这都不是我去学习迭代法的主要原因,毕竟递归法真的太简单明了了,让我学习迭代法的原因是这段评论。

 看到了吧,兄弟们,递归法太简单了,面试谁考你递归遍历呀,所以迭代还是很有必要学习学习的,在这里用力扣的代码重新给大家介绍递归法,同时补充迭代法。

前序遍历

链接:144. 二叉树的前序遍历 - 力扣(LeetCode)

递归法

class Solution {
    List<Integer> list = new ArrayList<>();
    private void preOrder(TreeNode root) {
        if (root==null) {
            return;
        } else {
            list.add(root.val);
            preOrder(root.left);
            preOrder(root.right);
        }
    }
    public List<Integer> preorderTraversal(TreeNode root) {
        preOrder(root);
        return list;
    }
}

此时的递归法显得那么的朴实无华,简洁明了,这里我们把打印节点换成了向List中添加元素,思路很简单,我们先添加元素然后preOrder(root.left)调用自己开始递归,直到我们的节点为空节点,此时直接返回然后调用preOrder(root.right)接着进行递归。

迭代法

class Solution {
    public List<Integer> preorderTraversal(TreeNode root) {
        List<Integer> lis = new LinkedList<>();
        Stack<TreeNode> stack = new Stack();
        while (!stack.isEmpty() || root!=null) {
            while (root != null) {
                lis.add(root.val);
                stack.push(root);
                root = root.left;
            }
            root = stack.pop();
            root = root.right;
        }
        return lis;
    }
}

迭代法的话就稍微复杂一点了,得用到我们的栈来实现了,先new一个List用于储存我们的数据,然后再new一个栈出来辅助我们来遍历二叉树,因为是先根遍历所以我们只要进了循环直接add向里面添加元素,然后进行压栈操作,将我们的根节点给压进去,然后root==root.left(根完了之后就是左)。

 如图接着就是将栈顶元素A给出了,然后就找到A的右,A的右为空所以继续pop()就找到了B,依此类推直到我们的栈为空,但是我们的第一个循环里面加了一个root!= null,这是因为刚开始我们的栈是空的,不然刚开始我们进不去循环。

中序遍历

递归法

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

与先序遍历相比,中序遍历只需要将add(0的位置改为在root.left和root.right的中间就行了。

迭代法

class Solution {
    public List<Integer> inorderTraversal(TreeNode root) {
        List<Integer> list = new ArrayList<>();
        Stack<TreeNode> stack = new Stack<>();
        while (root!=null || !stack.isEmpty()) {
            while (root!=null) {
                stack.push(root);
                root = root.left;
            }
            root = stack.pop();
            list.add(root.val);
            root = root.right;
        }
        return list;
    }
}

与先序遍历相比中序遍历的迭代也只是改变了add()的位置,因为只有我们的左完了也就是我们的内部的while()循环结束了,我们在弹出元素的时候才能add()同时抢在root=root.right的前面。

后序遍历

递归法

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

后序遍历的递归法同上我们只需要把add()放在root.right的后面就行了。

迭代法

class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        List<Integer> list = new ArrayList<>();
        Stack<TreeNode> stack = new Stack<>();
        TreeNode prev = null;
        while (root!=null || !stack.isEmpty()) {
            while (root!=null) {
                stack.push(root);
                root = root.left;
            }
            TreeNode ret = stack.peek();
            if (ret.right != null && ret.right != prev) {
                root = ret.right;
            } else {
                list.add(ret.val);
                stack.pop();
                prev = ret;
            }
            
        }
        return list;
    }
}

后序遍历的迭代法相对而言就比较麻烦一点,因为我们不能参考前面直接的把add()加在root.right的后面这样的话我们的原root就被弹出去了,然后也不会再次遍历它了,我们找到的只是root的right。

所以这时候我们用了一个ret用于接受我们的栈顶元素,如果我们栈顶元素的所在的节点的右为空的话,因为我们是左右根遍历的,既然右没了那就不需要再去找右了直接把当前的左给list.add()了,如果有右的话就接着 root=root.right 往右传接着探索,这时我们要注意一个bug。

 当遇到这种情况的时候,当我们3节点弹出的时候会对F进行操作然后F节点的右不为空又要push()3这个节点,然后就会陷入死循环,这时候就会凸显出来我们prve的作用了,记录上一个弹出的节点只有我们roo.right的节点不是prev我们才会接着遍历它,否则就弹出,同时我们弹出的时候要用prev标记一下这个节点。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/3534.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

渗透测试 | 域名信息收集

0x00 前言 信息收集可以说是在渗透测试中最重要的一部分&#xff0c;上文对 IP 信息收集做了一个简要的叙述&#xff0c;认识了 CDN 技术和网络空间搜索引擎。但是很多网站的主站因为访问流量过大的原因通常会使用 CDN 技术&#xff0c;同时也可以有效防止 DDOS 攻击。在域名信…

前端面经 强缓存与协商缓存

前端面经 强缓存与协商缓存 图片多来自第三方平台 文章目录前端面经 强缓存与协商缓存适用场合浏览器缓存的过程缓存规则强缓存&#xff08;本地缓存&#xff09;协商缓存缓存分为两种&#xff1a;强缓存和协商缓存&#xff0c;根据响应的header内容来决定 获取资源形式状态码…

【C++】二叉搜索树

前言 hi~大家好呀&#xff0c;欢迎点进我的C学习笔记~ 我的前一篇C笔记链接~ 【C】多态_柒海啦的博客-CSDN博客 本篇需要用到的基础二叉树C语言实现链接~ 用c语言实现一个简单的链表二叉树_柒海啦的博客-CSDN博客_c语言建立二叉树链表 我们知道&#xff0c;查找一个数据会有很多…

数据库基本操作

目录 数据库操作 创建数据库 查看数据库 选择数据库 删除数据库 注释 数据表操作 创建数据表 查看数据表 查看数据表 查看数据表的相关信息 修改数据表 修改数据表名称 修改表选项 查看表结构 查看数据表的字段信息 查看数据表的创建信息 查看数据表结构 修…

linux进程间通信之共享内存

目录 一&#xff0c;共享内存原理 二&#xff0c;创建共享内存 1&#xff0c;shmget创建共享内存 2&#xff0c;shmat挂接共享内存 3&#xff0c;shmdt取消挂接共享内存 4&#xff0c;shmctl删除共享内存 三&#xff0c;代码使用 1,com.hpp 2&#xff0c;ipc_client.c…

Allegro基本规则设置指导书之Physical Region

Allegro基本规则设置指导书之Physical Region 下面介绍基本规则设置指导书之Physical Region 空白的地方创建一个Region 给新建的Region匹配一个规则,所有区域里面的Physical相关的都按照Region的规则来 当部分网络想按照本身的规则来匹配,可以创建region-Class 然后匹配…

目标检测算法——医学图像开源数据集汇总(附下载链接)

关注”PandaCVer“公众号 深度学习Tricks&#xff0c;第一时间送达 目录 1.血细胞图像数据 2.眼病深度学习数据集 3.皮肤病数据集 4.膝关节 X 射线图像数据集 小海带整理不易&#xff0c;小伙伴们记得一键三连喔&#xff01;&#xff01;&#xff01; >>>一起交流…

VisualSVN 是 Visual Studio 的专业级 Subversion 集成插件

用于 Visual Studio 的 VisualSVN 专业且无缝的 Subversion 集成。 专业级 Subversion 集成 VisualSVN 是 Visual Studio 的专业级 Subversion 集成插件。 VisualSVN 的主要优点是&#xff1a; 无与伦比的可靠性&#xff1a; Visual Studio 永远不会因为 VisualSVN 而崩溃或挂…

保护鲸鱼动物网页设计作业 静态HTML宠物主题网页作业 DW鲸鱼网站模板下载 大学生简单动物网页作品代码 个人网页制作 学生个人网页

&#x1f389;精彩专栏推荐 &#x1f4ad;文末获取联系 ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 &#x1f482; 作者主页: 【主页——&#x1f680;获取更多优质源码】 &#x1f393; web前端期末大作业&#xff1a; 【&#x1f4da;毕设项目精品实战案例 (10…

2022阿里云栖大会,顶尖科技趋势峰会和全链路元宇宙体验

2022年的11月3日-5日是阿里巴巴云栖大会的日子&#xff0c;地点在云栖小镇&#xff0c;本人有幸报名参加了5日那场&#xff0c;因为5日是周六。秉着打工人工作日需要搬砖&#xff0c;因为“公司离不开我”&#xff0c;哈哈哈&#xff0c;实际上是每天满满的工作量。所以只能选择…

一文彻底搞懂协程(coroutine)是什么,值得收藏

什么是协程 我们可以简单的认为&#xff1a;协程就是用户态的线程&#xff0c;但是上下文切换的时机是靠调用方&#xff08;写代码的开发人员&#xff09;自身去控制的。 同时&#xff0c;协程和用户态线程非常接近&#xff0c;用户态线程之间的切换不需要陷入内核&#xff0…

NYIST(计科ACMTC)第三次招新赛题解

A题 原文, 原比赛B题 牛客练习赛104【出题人题解】 - 知乎 直接输出 输入的数 就可以了 B题 C题 找到分别处理"无留陀的化身"坐标轴的x轴和y轴, 组合成无留陀的坐标, 再遍历求纳西妲的坐标, 相减即可 /* ⣿⣿⣿⣿⣿⣿⡷⣯⢿⣿⣷⣻⢯⣿⡽⣻⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿…

ROS小工具学习与使用

ROS小工具学习与使用 rqt的使用 rqt_bag工具 rqt_bag <your bagfile> #使用rqt_bag查看你的rosbag例如&#xff1a;可以查看第一帧GPS的rawdata信息&#xff0c;如下图&#xff1a; 参考文献&#xff1a; 1、http://wiki.ros.org/rqt_bag 2、rosbag与rqt_bag的常用 rq…

Nacos学习笔记

视频学习指路&#xff1a; 【SpringCloudRabbitMQDockerRedis搜索分布式&#xff0c;系统详解springcloud微服务技术栈课程|黑马程序员Java微服务】 Nacos nacos注册中心的搭建 1.下载nacos的安装包&#xff0c;github地址&#xff1a;https://github.com/alibaba/nacos&…

Tkinter保姆级教程(上)

目录 什么是GUI Tkinter用法详解 第一个Tkinter程序 常用控件和属性 主窗口 Label标签控件 Button按钮控件 Entry输入控件 基本属性 Text 文本控件 列表框(ListBox)和组合框(Combobox) 单选框(Radiobutton)和多选框按钮(Checkbutton) 什么是GUI 图形用户界面&#x…

运算放大器正反馈负反馈判别法

---------------------------------------------------------------------------------------------------------------- 反馈可分为负反馈和正反馈。前者使输出起到与输入相反的作用&#xff0c;使系统输出与系统目标的误差减小&#xff0c;系统趋于稳定&#xff1b;后者使输出…

java面向对象(上)

一、java面向对象学习的三条主线1.java类以及类的成员&#xff1a;属性、方法、构造器&#xff1b;代码块、内部类。2.面向对象的三大特征&#xff1a;封装性&#xff0c;继承性&#xff0c;多态性&#xff0c;&#xff08;抽象性&#xff09;。3.其他关键字&#xff1a;this&a…

.net技术第一章

文章目录.NETC# (C Sharp)的特点C# 的应用范围.NET Framework1.2 创建简单的C#程序结构和书写规则类型的声明和使用类型的声明和使用命名空间使用方法命名空间举例注释Main方法命令行参数Main返回值控制台输入和输出例子格式化.NET 由微软公司提供的免费、跨平台的开源通用开发…

复杂分数 马蹄集

复杂分数 难度&#xff1a;白银 0时间限制&#xff1a;1秒 巴占用内存&#xff1a;64M 编写程序连续输入a1、a2、、a5,计算下列表达式的值并输出。本题不考虑输 入0&#xff0c;负数或者其他特殊情况。 1十1中 al 3 4 格式 输入格式&#xff1a;输入整型&#xff0c;空格分隔。…

Go语言学习(二) 函数

文章目录函数go函数基本语法go不支持重载go中支持可变参数函数 go函数基本语法 先来看看go中函数的基本使用 package mainimport "fmt"/* func 函数名(形参列表) (返回值类型列表) {执行语句..return 返回值列表 } */ //自定义函数&#xff1a;功能&#xff1a;两…