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;
                }

            }
        }
    }

}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

RabbitMq整合Springboot超全实战案例+图文演示+源码自取

目录 介绍 简单整合 简单模式 定义 代码示例 work模式 定义 代码示例 pubsub模式 定义 代码示例 routing模式 定义 代码示例 top模式 定义 代码 下单付款加积分示例 介绍 代码 可靠性投递示例 介绍 代码 交换机投递确认回调 队列投递确认回调 ​延迟消…

创作涌动·CSDN·21天创作挑战赛·第三期,正式开启报名!

​ 文章目录 ⭐️ 活动介绍⭐️ 活动详情⭐️ 活动奖品⭐️ 活动流程⭐️ 评审规则⭐️ 报名&投稿事项⭐️ 关于活动组织 活动报名地址&#xff08;点击跳转&#xff09; 本次活动与官方活动及其他博主的创作型活动并不不冲突&#xff01; ⭐️ 活动介绍 亲爱的小伙伴们&a…

树莓派Python程序开机自启动(Linux下Python程序开机自启动)

前一阵子用python编写了一个驱动I2C程序读写屏幕&#xff0c;输出IP的小程序&#xff0c;程序编好后需要树莓派使能程序开机自启动。其实这些方法对任何Linux系统都适用。 方法一&#xff1a;此方法的缺点是不进入默认pi的账号&#xff0c;甚至不开hdmi开启桌面的话&#xff0…

连夜整理的6个开源项目,都很实用

偶然找到的这个宝藏网站&#xff0c;站内集齐了大量的开源项目。 推荐实用的项目 1、vueNextAdmin 基于 vue3.x CompositionAPI setup 语法糖 typescript vite element plus vue-router-next pinia 技术&#xff0c;适配手机、平板、pc 的后台开源免费模板&#xff0c;…

使用K-means把人群分类

1.前言 K-mean 是无监督的聚类算法 算法分类&#xff1a; 2.实现步骤 1.数据加工&#xff1a;把数据转为全数字&#xff08;比如性别男女&#xff0c;转换为0 和 1&#xff09; 2.模型训练 fit 3.预测 3.代码 原数据类似这样(source&#xff1a;http:img-blog.csdnimg.cn…

06 数仓平台MaxWell

Maxwell简介 Maxwell是由Zendesk公司开源&#xff0c;用 Java 编写的MySQL变更数据抓取软件&#xff0c;能实时监控 MySQL数据库的CRUD操作将变更数据以 json 格式发送给 Kafka等平台。 Maxwell输出数据格式 Maxwell 原理 Maxwell工作原理是实时读取MySQL数据库的二进制日志…

Kubernetes(K8s)数据存储-09

数据存储 在前面已经提到&#xff0c;容器的生命周期可能很短&#xff0c;会被频繁地创建和销毁。那么容器在销毁时&#xff0c;保存在容器中的数据也会被清除。这种结果对用户来说&#xff0c;在某些情况下是不乐意看到的。为了持久化保存容器的数据&#xff0c;kubernetes引…

idea利用spring框架整合thymeleaf展现数据库数据

idea初步利用thymeleaf展现列表 上一篇文章简单展现自己写的列表&#xff1b; 这篇文章连接mysql数据库实现数据库数据展现 主要三个文件 controller指定html界面 package com.example.appledemo.controller;import com.example.appledemo.mapper.UserMapper; import com.exam…

“名创优品小动物保护公益基金”项目成立,捐赠1000万助力美好生活

近日&#xff0c;名创优品宣布捐赠1000万元成立“名创优品小动物保护公益基金”项目&#xff0c;将通过专业、关爱、共生的公益理念和行动&#xff0c;助力构建良好的社区生态和美好生活方式&#xff0c;与年轻一代探索中国公益创新发展。 名创优品捐赠1000万元成立“名创优品小…

操作系统概论:揭秘计算机背后的神秘力量

操作系统概论 & 功能 概述定义操作系统功能作为系统资源的管理者向上层提供方便易用的服务作为最接近硬件的层次 主页传送门&#xff1a;&#x1f4c0; 传送 概述 概念&#xff1a; 定义 控制和管理计算机硬件和软件资源的程序一种系统软件为上层用户、应用程序提供简单易…

钉钉员工组织资料实时同步至飞书的应用解析

如何实现应用之间的同步&#xff1f; 随着企业应用的日益增多&#xff0c;在帮助企业提供办公效率的同时&#xff0c;也增加了对这些应用的运维成本。有没有一种好的办法&#xff0c;实现saas应用之间的桥梁搭建&#xff0c;自动化地完成不同应用之间的数据流转呢&#xff1f;…

C语言枚举详解,typedef简介(能看懂文字就能明白系列)

系列文章目录 C语言基础专栏 笔记详解 &#x1f31f; 个人主页&#xff1a;古德猫宁- &#x1f308; 信念如阳光&#xff0c;照亮前行的每一步 文章目录 系列文章目录&#x1f308; *信念如阳光&#xff0c;照亮前行的每一步* 前言一、枚举类型的声明枚举常量三、枚举类型的优…

智能指针及强相关知识经验总结 --- 移动语义、引用计数、循环引用、move()、自定义删除器等

目录 前言 一、shared_ptr 1. 基本用法和构造方法 2. 引用计数机制 3. weak_ptr 解决循环引用 二、unique_ptr 1. 基本用法和构造方法 2. 独占性 3. 所有权转移 1&#xff09;unique_ptr :: release() 2&#xff09;移动语义 和 move() 三、 对比 shared_ptr 和 un…

Dropwizard-metric的使用

背景 近期在开发中用到了dropwizard-metric作为监控metric的埋点框架&#xff0c;由于是分布式的系统&#xff0c;前期曾经对比过hadoop-metric的实现和dropwizard-metric的实现&#xff0c;因为开发的项目后续会和hadoop的项目有一定的上下游关系&#xff0c;所以考虑排除掉h…

Re 花指令学习

概念 花指令又名垃圾代码、脏字节&#xff0c;英文名是junk code。花指令就是在不影响程序运行的情况下&#xff0c;往真实代码中插入一些垃圾代码&#xff0c;从而影响反汇编器的正常运行&#xff1b;或是起到干扰逆向分析人员的静态分析&#xff0c;增加分析难度和分析时间。…

开发的客户收到样品表示质量不如原供应商如何应对

有小伙伴问&#xff0c;在开发客户的过程当中&#xff0c;给客户寄了样品&#xff0c;客户说他的样品没有原来供应商的好怎么办&#xff1f; 这个问题我们来想一下&#xff0c;客户既然愿意把地址给我们&#xff0c;愿意去接你的样品&#xff0c;说明什么&#xff1f;说明客户…

系列十五、SpringBoot的启动原理分析

一、概述 所谓SpringBoot的启动原理&#xff0c;翻译成大白话就是"当我们在主启动类上运行run方法时&#xff0c;SpringBoot底层到底做了什么事情&#xff0c;能够帮助我们启动一个Spring的web应用"&#xff0c;上边用大白话解释了一下什么是SpringBoot的启动原理&am…

PyQt6 QTabWidget选项卡控件

​锋哥原创的PyQt6视频教程&#xff1a; 2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~共计37条视频&#xff0c;包括&#xff1a;2024版 PyQt6 Python桌面开发 视频教程(无废话…

0001微信报Dynamic.dll,微信报DynamicDll64.dll 环境

0001微信报Dynamic.dll&#xff0c;微信报DynamicDll64.dll 1. 环境 操作系统&#xff1a;Windows10专业版 2. 现象 某赛通加密客户端升级后&#xff0c;企业微信和微信报如下错误 分析 该加密软件升级导致的异常 3. 解决办法 企业微信&#xff1a;粘贴DynamicDll.dll到…

ULAM公链第九十六期工作总结

迈入12月&#xff0c;接下来就是雪花&#xff0c;圣诞&#xff0c;新年和更好的我们&#xff01;愿生活不拥挤&#xff0c;笑容不必刻意&#xff0c;愿一切美好如期而至&#xff01; 2023年11月01日—2023年12月01日关于ULAM这期工作汇报&#xff0c;我们通过技术板块&#xff…