Java 数据结构篇-二叉树的深度优先遍历(实现:递归方式、非递归方式)

🔥博客主页: 【小扳_-CSDN博客】
❤感谢大家点赞👍收藏⭐评论✍
  

文章目录

        1.0 二叉树的说明

        1.1 二叉树的实现

        2.0 二叉树的优先遍历说明

        3.0 用递归方式实现二叉树遍历

        3.1 用递归方式实现遍历 - 前序遍历

        3.2 用递归方式实现遍历 - 中序遍历

        3.3 用递归方式实现遍历 - 后序遍历

        4.0 用非递归方式实现二叉树遍历

        4.1 用非递归方式实现遍历 - 前序遍历

        4.2 用非递归方式实现遍历 - 中序遍历

        4.3 用非递归方式实现遍历 - 后序遍历

        5.0 深度遍历的完整代码


        1.0 二叉树的说明

        二叉树是一种树形数据结构,其中每个节点最多有两个子节点,分别称为左子节点和右子节点。二叉树可以为空,或者包含一个根节点和两个指向左子树和右子树的指针。

        1.1 二叉树的实现

二叉树可以是空的,也可以是具有以下特点的非空树:

         - 每个节点最多有两个子节点,分别称为左子节点和右子节点

         - 左子树和右子树都是二叉树

代码实现如下:

public class MyBinaryTree {//指向左子树private MyBinaryTree left;//当前节点的值private int val;//指向右子树private MyBinaryTree right;//构造方法一:带一个值的参数public MyBinaryTree(int val) {this.val = val;}//构造方法二:带三个参数public MyBinaryTree(MyBinaryTree left, int val, MyBinaryTree right) {this.left = left;this.val = val;this.right = right;}}

        实现两个构造方法是为了创建二叉树的时候会比较方便一些。

二叉树的创建如下:

MyBinaryTree myBinaryTree = new MyBinaryTree(new MyBinaryTree(new MyBinaryTree(4),2,null),1,new MyBinaryTree(new MyBinaryTree(5),3,new MyBinaryTree(6)));

该二叉树的图片:

        2.0 二叉树的优先遍历说明

        二叉树的优先遍历是指按照一定顺序访问二叉树中的所有节点。常见的三种优先遍历方式包括:前序遍历、中序遍历和后序遍历。可以使用递归实现非递归实现这三种遍历方式。

               

        3.0 用递归方式实现二叉树遍历

        递归实现前序遍历、中序遍历、后续遍历。用递归实现这三种遍历方式的区别就是对于数据什么时候处理,如打印。

                - 前序遍历(Preorder Traversal):首先访问根节点,然后递归地遍历左子树,最后递归地遍历右子树。在前序遍历中,节点的访问顺序是“根-左-右”。

                - 中序遍历(Inorder Traversal):首先递归地遍历左子树,然后访问根节点,最后递归地遍历右子树。在中序遍历中,节点的访问顺序是“左-根-右”。

                - 后序遍历(Postorder Traversal):首先递归地遍历左子树,然后递归地遍历右子树,最后访问根节点。在后序遍历中,节点的访问顺序是“左-右-根”。

        3.1 用递归方式实现遍历 - 前序遍历

        具体思路为:在递出之前需要对当前节点的值访问,然后再接着向左子树递出,一直下去,每一次都需要先对当前节点的值访问,直到 node == null 时,左子树结束递出。当右子树此时也为  node == null 时,从叶子节点开始回归,回归到上一个节点的右子树。

代码实现如下:

假设该二叉树的图:

那么前序遍历打印的顺序为:124356

    //使用递归实现前序遍历public void preTraversalRecursion(MyBinaryTree node) {if (node == null) {return;}System.out.print(node.val + " ");preTraversalRecursion(node.left);preTraversalRecursion(node.right);}

        

        3.2 用递归方式实现遍历 - 中序遍历

        具体思路为:向左子树递出,一直下去,直到 node == null 时,左子树结束递出。再来对当前节点的值进行访问,接着继续向着右子树递出,当右子树此时也为 node == null 时,从叶子节点开始回归,回归到上一个节点的右子树前先对当前节点的值进行访问。

代码实现如下:

假设该二叉树的图:

中序遍历打印的顺序为:421563

    //使用递归实现中序遍历public void middleTraversalRecursion(MyBinaryTree node) {if (node == null) {return;}middleTraversalRecursion(node.left);System.out.print(node.val + " ");middleTraversalRecursion(node.right);}

        相对于前序遍历,中序遍历就是将对节点的值的访问进行换了一下位置。先进行左子树递归,再来访问值,最后再右子树递归。

        3.3 用递归方式实现遍历 - 后序遍历

        具体思路为:向左子树递出,一直下去,直到 node == null 时,左子树结束递出。再接着继续向着右子树递出,再来对当前节点的值进行访问,当右子树此时也为 node == null 时,从叶子节点开始回归,回归到对当前节点的值进行访问。

代码实现如下:

假设该二叉树的图:

后续遍历打印顺序为:425631

    //使用递归实现后续遍历public void postTraversalRecursion(MyBinaryTree node) {if (node == null) {return;}postTraversalRecursion(node.left);postTraversalRecursion(node.right);System.out.print(node.val + " ");}

        4.0 用非递归方式实现二叉树遍历

        非递归实现前序遍历、中序遍历、后续遍历。用非递归实现这三种遍历方式的区别是访问的顺序不同,而前中后序它们所走的路线是一致的

假设该树的图片:

        具体思路为:先定义一个当前节点 curr 一开始指向根节点,还需要有一个栈的数据结构来存储数据。从根节点开始往后先左子树一直遍历下去 curr = curr.left ,每次都需要存储当前节点到栈中,直到 curr == null 时,说明了左子树已经遍历完毕了。此时需要栈弹出节点,来寻找 “来时的路”,弹出来的节点就是离当前节点最近的节点,在原路返回的时候,还要判断当前节点是否还有右子树,将弹出栈的节点的右子树赋值给 curr = tp.right 。如果右子树不为 null ,就会继续对当前节点的左子树遍历,且把当前节点压入栈中,方便记住 “来时的路”,;如果右子树不为 null ,继续把栈中的节点弹出,直到栈中没有了节点且 curr == null 时,因此整个二叉树就遍历结束了。

        4.1 用非递归方式实现遍历 - 前序遍历

        对于前序遍历来说,就是在来到下一个节点之前对当前节点的值先访问了。

代码如下:

    //非递归实现前序遍历public void preTraversal(MyBinaryTree node) {MyBinaryTree curr = node;LinkedList<MyBinaryTree> stack = new LinkedList<>();while (curr != null || !stack.isEmpty()) {if (curr != null) {System.out.print(curr.val + " ");stack.push(curr);curr = curr.left;}else {MyBinaryTree tp = stack.pop();curr = tp.right;}}System.out.println();}

        4.2 用非递归方式实现遍历 - 中序遍历

        对于中序遍历来说,在栈弹出节点后,对弹出的节点进行访问。

代码如下:

    //非递归实现中序遍历public void middleTraversal(MyBinaryTree node) {MyBinaryTree curr = node;LinkedList<MyBinaryTree> stack = new LinkedList<>();while ( curr != null || !stack.isEmpty()) {if (curr != null) {stack.push(curr);curr = curr.left;}else  {MyBinaryTree tp = stack.pop();System.out.print(tp.val + " ");curr = tp.right;}}System.out.println();}

        4.3 用非递归方式实现遍历 - 后序遍历

        相比前面两种前中后序遍历,后序遍历会多了一个判断,由于先访问完当前节点的左子树和右子树后,才能对当前节点的值进行访问,所以不能访问完左子树后,立马将当前节点弹出栈,还需要保留该节点,先对该节点的右节点进行访问完毕之后,再来访问该节点的值,最后就可以弹出栈了。需要先判断,当前节点的右子树为 null 时,可以弹出当前节点或者已经对当前节点的右节点访问完毕之后,也可以将当前节点弹出。

        那么如何来判断是否访问完毕当前节点的右子树了?

        可以用一个变量来标记,只需要判断最近弹出栈的节点是否为当前节点的右子树节点即可。

代码如下:

    //非递归实现后序遍历public void postTraversal(MyBinaryTree node) {MyBinaryTree crr = node;LinkedList<MyBinaryTree> stack = new LinkedList<>();MyBinaryTree pop = null;//最后一次弹栈元素while ( crr != null || !stack.isEmpty()) {if (crr != null) {stack.push(crr);crr = crr.left;}else {MyBinaryTree tp = stack.peek();if ( tp.right == null || tp.right == pop ) {pop = stack.pop();System.out.print(pop.val + " ");}else {crr = tp.right;}}}}

        5.0 深度遍历的完整代码

import java.util.LinkedList;public class MyBinaryTree {//指向左子树private MyBinaryTree left;//当前节点的值private int val;//指向右子树private MyBinaryTree right;//构造方法一:带一个值的参数public MyBinaryTree(int val) {this.val = val;}//构造方法二:带三个参数public MyBinaryTree(MyBinaryTree left, int val, MyBinaryTree right) {this.left = left;this.val = val;this.right = right;}//使用递归实现前序遍历public void preTraversalRecursion(MyBinaryTree node) {if (node == null) {return;}System.out.print(node.val + " ");preTraversalRecursion(node.left);preTraversalRecursion(node.right);}//使用递归实现中序遍历public void middleTraversalRecursion(MyBinaryTree node) {if (node == null) {return;}middleTraversalRecursion(node.left);System.out.print(node.val + " ");middleTraversalRecursion(node.right);}//使用递归实现后续遍历public void postTraversalRecursion(MyBinaryTree node) {if (node == null) {return;}postTraversalRecursion(node.left);postTraversalRecursion(node.right);System.out.print(node.val + " ");}//非递归实现前序遍历public void preTraversal(MyBinaryTree node) {MyBinaryTree curr = node;LinkedList<MyBinaryTree> stack = new LinkedList<>();while (curr != null || !stack.isEmpty()) {if (curr != null) {System.out.print(curr.val + " ");stack.push(curr);curr = curr.left;}else {MyBinaryTree tp = stack.pop();curr = tp.right;}}System.out.println();}//非递归实现中序遍历public void middleTraversal(MyBinaryTree node) {MyBinaryTree curr = node;LinkedList<MyBinaryTree> stack = new LinkedList<>();while ( curr != null || !stack.isEmpty()) {if (curr != null) {stack.push(curr);curr = curr.left;}else  {MyBinaryTree tp = stack.pop();System.out.print(tp.val + " ");curr = tp.right;}}System.out.println();}//非递归实现后序遍历public void postTraversal(MyBinaryTree node) {MyBinaryTree crr = node;LinkedList<MyBinaryTree> stack = new LinkedList<>();MyBinaryTree pop = null;//最后一次弹栈元素while ( crr != null || !stack.isEmpty()) {if (crr != null) {stack.push(crr);crr = crr.left;}else {MyBinaryTree tp = stack.peek();if ( tp.right == null || tp.right == pop ) {pop = stack.pop();System.out.print(pop.val + " ");}else {crr = tp.right;}}}}}

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

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

相关文章

逻辑回归 使用Numpy实现逻辑回归

使用Numpy实现逻辑回归 sigmoid 函数 g ( z ) 1 ( 1 e − z ) g(z)\frac{1}{(1e^{−z} )} g(z)(1e−z)1​ # sigmoid 函数 def sigmod(z):return 1/(1np.exp(-z))线性计算与梯度下降 J ( θ ) − 1 m [ ∑ i 1 m y ( i ) l o g ⁡ ( h θ ( x ( i ) ) ) ( 1 − y ( i ) …

ROS 元功能包

ROS元功能包&#xff08;Metapackage&#xff09;是一种特殊的软件包&#xff0c;它本身并不包含任何可执行代码或数据文件。在ROS 1中&#xff0c;可以通过catkin_create_pkg命令创建元功能包。 相反&#xff0c;它的主要目的是作为一组相关功能包的集合或者依赖关系列表。使…

国标GB28181视频监控EasyCVR内网环境部署无法启动怎么办?

安防视频监控系统EasyCVR平台可拓展性强、视频能力灵活、部署轻快&#xff0c;可支持的主流标准协议有国标GB28181、RTSP/Onvif、RTMP等&#xff0c;以及支持厂家私有协议与SDK接入&#xff0c;包括海康Ehome、海大宇等设备的SDK等&#xff0c;能对外分发RTMP、RTSP、HTTP-FLV、…

微信聊天窗口测试用例

以前没测过客户端的测试&#xff0c;昨天面试被问到聊天窗口测试场景设计&#xff0c;感觉自己答的不好&#xff0c;结束后上网查了一下客户端/app测试的要点&#xff0c;按照测试策略来分&#xff0c;主要涉及到如下测试类型&#xff1a; 1、功能测试 2、性能测试 3、界面测试…

GS016电动工具调速控制电路芯片,7V ~ 24V 7mA ~ 10mA具 有电源电压范围宽、功耗小、抗干扰能力强等特点

GS016是一款直流有刷电机调速电路&#xff0c;输出端内置14V钳位结构&#xff0c;具 有电源电压范围宽、功耗小、抗干扰能力强等特点。通过桥接内部电阻网 络&#xff0c;可以改变PWM占空比输出&#xff0c;达到控制电机转速作用。采用SOP14的封装形式封装。 主要特点&#xf…

Hadoop学习笔记(HDP)-Part.10 创建集群

目录 Part.01 关于HDP Part.02 核心组件原理 Part.03 资源规划 Part.04 基础环境配置 Part.05 Yum源配置 Part.06 安装OracleJDK Part.07 安装MySQL Part.08 部署Ambari集群 Part.09 安装OpenLDAP Part.10 创建集群 Part.11 安装Kerberos Part.12 安装HDFS Part.13 安装Ranger …

ipad Google浏览器,使用默认搜索,页面使用pc模式

ipad Google浏览器&#xff0c;使用默认搜索&#xff0c;页面使用pc模式 1. 设置默认搜索引擎 2. 设置页面使用PC模式 参考&#xff1a;https://zhuanlan.zhihu.com/p/556041670

从零开始学习 JS APL(五):完整指南和实例解析

目录 学习目标&#xff1a; 学习内容&#xff1a; 学习时间&#xff1a; 学习内容&#xff1a; Window对象&#xff1a; 定时器-延时函数&#xff1a; JS 执行机制&#xff1a; location对象&#xff1a; 本地存储&#xff1a; 本地存储分类- localStorage&#xff1a…

OTN设备,ZXONE 9700,ZXMP M721

文章目录 ZXONE 9700分组OTN产品产品特点 ZXMP M721城域边缘OTN产品产品特点 ZXONE 9700分组OTN产品 ZXONE 9700系列产品&#xff0c;支持10G/40G/100G/400G传输速率&#xff0c;可实现28.8T/14.4T/9.2T/4.4T ODUk的大容量电层交叉和10G/40G/100G/400G波长的光层交叉及分组交换…

WordPress免费插件大全清单【2023最新】

WordPress已经成为全球范围内最受欢迎的网站建设平台之一。要让您的WordPress网站更具功能性、效率性&#xff0c;并提供卓越的用户体验&#xff0c;插件的选择与使用变得至关重要。 WordPress插件的作用 我们先理解一下插件在WordPress生态系统中的作用。插件是一种能够为Wo…

【云原生-K8s】检查yaml文件安全配置kubesec部署及使用

基础介绍基础描述特点 部署在线下载百度网盘下载安装 使用官网样例yamlHTTP远程调用安全建议 总结 基础介绍 基础描述 Kubesec 是一个开源项目&#xff0c;旨在为 Kubernetes 提供安全特性。它提供了一组工具和插件&#xff0c;用于保护和管理在 Kubernetes 集群中的工作负载和…

前端三段面试经历(字节、滴滴、虎牙),讲真的还挺难的

小编整理最近了小伙伴中具有代表性的面试经历及面试题&#xff0c;他们的情况小编就不多说了&#xff0c;一起看看他们面试中都遇到了什么问题。 字节一面 刚开始就问我的项目&#xff0c;问得很详细。 webpack 提高构建速度的方式loader 输入什么产出什么 &#xff1f;webpac…

全光谱护眼灯好吗?考公用的台灯推荐

全光谱护眼台灯是近年来新兴的一种照明产品&#xff0c;其使用全光谱LED灯泡&#xff0c;能够模拟自然光的光谱&#xff0c;为人们提供更加舒适、柔和的照明环境。相比传统的白炽灯或荧光灯&#xff0c;全光谱护眼台灯在照明效果、节能环保、眼睛健康等方面都有着明显的优势。 …

mysql:免费的GUI客户端工具推荐并介绍常用的操作

给大家推荐几个常用的 mysql 数据库客户端 sequel-pro sequel-ace 官网下载地址 免费 sequel-ace 可以理解为 Sequel Pro 的升级版&#xff0c;由于Sequel Pro官方不维护了&#xff0c;特别是对 MySQL 8.0 支持不好&#xff0c;所以现在由社区维护了新分支 sequel-ace&#x…

生鲜蔬果展示预约小程序作用是什么

线下生鲜蔬果店非常多&#xff0c;对商家来说主要以同城生意为主&#xff0c;而在互联网电商的发展下&#xff0c;更多的商家会选择搭建私域商城进行多渠道的销售卖货和拓展&#xff0c;当然除了直接卖货外&#xff0c;还有产品纯展示或预约订购等需求。 但无论哪种模式&#…

【云原生 | Docker】Docker核心概念 应用上手最佳流程

&#x1f935;‍♂️ 个人主页: AI_magician &#x1f4e1;主页地址&#xff1a; 作者简介&#xff1a;CSDN内容合伙人&#xff0c;全栈领域优质创作者。 &#x1f468;‍&#x1f4bb;景愿&#xff1a;旨在于能和更多的热爱计算机的伙伴一起成长&#xff01;&#xff01;&…

面试题:Java Switch 是如何支持 String 的,为什么不支持 long

目录 前言一、结论二、枚举类型是咋变成 int 类型的&#xff1f;三、String 类型是咋变成 int 类型的&#xff1f;四、它们的包装类型支持吗&#xff1f; 前言 我们知道Java Switch 支持byte、short、int类型&#xff0c;在JDK 1.5 时&#xff0c;支持了枚举类型&#xff0c;在…

STC15F100E单片机模拟串口

文章目录 一、芯片简介二、开发环境三、软件模拟串口参考 一、芯片简介 STC15F100系列单片机是宏晶科技生产的单时钟/机器周期(1T)的单片机&#xff0c;新一代8051单片机&#xff0c;指令代码完全兼容传统8051&#xff0c;但是速度快6-12倍。 内部集成R/C时钟&#xff0c;5MHz…

25、pytest的测试报告插件allure

allure简介 在这里&#xff0c;你将找到使用allure创建、定制和理解测试报告所需的一切。开始让你的测试沟通更清晰&#xff0c;更有影响力。 Allure Report是一个实用程序&#xff0c;它处理由兼容的测试框架收集的测试结果并生成HTML报告。 安装allure 1、确保安装了Java…

【SSM源码】基于JAVA的高校竞赛和考级查询系统

该项目含有源码、文档、PPT、配套开发软件、软件安装教程、项目发布教程等学习内容。 目录 一、项目介绍&#xff1a; 二、文档学习资料&#xff1a; 三、模块截图&#xff1a; 四、开发技术与运行环境&#xff1a; 五、代码展示&#xff1a; 六、数据库表截图&#xff1a…