【数据结构】二叉数的存储与基本操作的实现

文章目录

  • 🍀二叉树的存储
  • 🌳二叉树的基本操作
    • 🐱‍👤二叉树的创建
    • 🐱‍👓二叉树的遍历
      • 🎡前中后序遍历
        • 📌前序遍历
        • 📌中序遍历
        • 📌后续遍历
      • 🛫层序遍历
      • 🐱‍👤前中后序代码实现(递归)
        • 🚩前序遍历
        • 🚩中序遍历
        • 🚩后续遍历
      • 🛬前中后序练习题
    • 🐱‍🏍二叉树的基本操作
      • 🎈获取树中节点的个数
      • 🎈获取叶子节点的个数
      • 🎈获取第K层节点的个数
      • 🎈 获取二叉树的高度
      • 🎈检测值为value的元素是否存在
  • ⭕总结

在这里插入图片描述

🍀二叉树的存储

二叉树的存储结构分为:顺序存储和类似于链表的链式存储

这里博主讲一下链式存储

二叉树的链式存储是通过一个一个的节点引用起来的,常见的表示方式有二叉和三叉表示方式

二叉表示:

// 孩子表示法
class Node {
	int val; // 数据域
	Node left; // 左孩子的引用,常常代表左孩子为根的整棵左子树
	Node right; // 右孩子的引用,常常代表右孩子为根的整棵右子树
}

三叉表示:

/
/ 孩子双亲表示法
class Node {
	int val; // 数据域
	Node left; // 左孩子的引用,常常代表左孩子为根的整棵左子树
	Node right; // 右孩子的引用,常常代表右孩子为根的整棵右子树
	Node parent; // 当前节点的根节点
}

这里博主主要讲解一下孩子表示法

🌳二叉树的基本操作

🐱‍👤二叉树的创建

在学习二叉树的基本操作前,需先要创建一棵二叉树,然后才能学习其相关的基本操作。由于现在大家对二叉树结构掌握还不够深入,为了降低大家学习成本,此处手动快速创建一棵简单的二叉树。
在这里插入图片描述

创建如下:

public class BinaryTree{
    public static class BTNode{
        BTNode left;
        BTNode right;
        int value;
        BTNode(int value){
            this.value = value;
        }
    }
    private BTNode root;
    public void createBinaryTree(){
        BTNode node1 = new BTNode(1);
        BTNode node2 = new BTNode(2);
        BTNode node3 = new BTNode(3);
        BTNode node4 = new BTNode(4);
        BTNode node5 = new BTNode(5);
        BTNode node6 = new BTNode(6);
        root = node1;
        node1.left = node2;
        node2.left = node3;
        node1.right = node4;
        node4.left = node5;
        node5.right = node6;
    }
}

注意:上述代码并不是创建二叉树的方式,真正创建二叉树方式后序详解重点讲解

在我们对二叉树进行基本操作之前,我们的先来回顾以下二叉树

二叉树是:

  1. 空树
  2. 非空:根节点,根节点的左子树、根节点的右子树组成的
    在这里插入图片描述
    从概念中可以看出,二叉树的每一个子树又是一个新的二叉树,所以可以知道二叉树定义是递归式的,因此后序基本操作中基本都是按照该概念实现的。

🐱‍👓二叉树的遍历

🎡前中后序遍历

学习二叉树结构,最简单的方式就是遍历。所谓遍历(Traversal)是指沿着某条搜索路线,依次对树中每个结点均做一次且仅做一次访问。访问结点所做的操作依赖于具体的应用问题(比如:打印节点内容、节点内容加1)

遍历是二叉树上最重要的操作之一,是二叉树上进行其它运算之基础

在遍历二叉树时,如果没有进行某种约定,每个人都按照自己的方式遍历,得出的结果就比较混乱,如果按照某种规则进行约定,则每个人对于同一棵树的遍历结果肯定是相同的。如果N代表根节点,L代表根节点的
左子树,R代表根节点的右子树,则根据遍历根节点的先后次序有以下遍历方式:

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

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

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

📌前序遍历

前序遍历(Preorder Traversal 亦称先序遍历)规则为

访问根结点—>根的左子树—>根的右子树

比如以下二叉树:
在这里插入图片描述
前序遍历的顺序为:

1.== 遍历A结点==
2. 遍历A结点的左子树
3. 遍历B结点
4. 遍历B结点的左子树
5. 遍历D结点
6. 判断D的左右子树为空后返回
7. 遍历B结点的右子树,为空返回
8. 此时A结点左子树遍历完,开始遍历A结点右子树
9. 遍历C结点
10.遍历C结点的左子树
11.遍历E结点
12.判断E结点的左右子树为空后返回
13.遍历C结点的右子树
14.遍历F结点
15.判断F结点的左右子树为空后返回
16.自此遍历完毕全部返回

最后前序的遍历结果为:

A->B->D->C->E->F

📌中序遍历

中序遍历(Inorder Traversal)的访问规则为:

根的左子树—>根节点—>根的右子树。

比如以下二叉树:

在这里插入图片描述
中序遍历的顺序为:

  1. 遍历A结点的左子树
  2. 遍历B结点的左子树
  3. 遍历D结点的左子树,发现为空返回
  4. 遍历D结点
  5. 遍历D结点的右子树,发现为空返回
  6. 遍历B结点
  7. 遍历B结点的右子树。发现为空返回
  8. 此时左子树遍历完成
  9. 遍历A结点
  10. 遍历A结点右子树
  11. 遍历C结点左子树
  12. 遍历E结点的左子树,发现为空返回
  13. 遍历E结点
  14. 遍历E结点的右子树,发现为空返回
  15. 遍历C结点
  16. 遍历C结点的左子树
  17. 遍历F结点的左子树,发现为空返回
  18. 遍历F结点
  19. 遍历F结点的右子树,发现为空返回
  20. 自此遍历完毕全部返回

最后中序的遍历结果为:

D->B->A->E->C->F

📌后续遍历

后序遍历(Postorder Traversal)的访问规则为:

根的左子树—>根的右子树—>根节点

比如以下二叉树:
在这里插入图片描述
后续遍历的顺序为:

  1. 遍历A结点的左子树
  2. 遍历B结点的左子树
  3. 遍历D结点的左子树,为空后返回
  4. 遍历D结点的右子树,为空后返回
  5. 遍历D结点
  6. 遍历B结点的右子树,为空后返回
  7. 遍历B结点
  8. 遍历A结点的右子树
  9. 遍历C结点的左子树
  10. 遍历E结点的左子树,为空后返回
  11. 遍历E结点的右子树,为空后返回
  12. 遍历E结点
  13. 遍历C结点的右子树
  14. 遍历F结点的左子树,为空后返回
  15. 遍历F结点的右子树,为空后返回
  16. 遍历F结点
  17. 遍历C结点
  18. 遍历A结点
  19. 至此遍历完毕,全部返回

最后后序的遍历结果为:

D->B->E->F->C->A

🛫层序遍历

层序遍历:除了先序遍历、中序遍历、后序遍历外,还可以对二叉树进行层序遍历。

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

在这里插入图片描述

🐱‍👤前中后序代码实现(递归)

我们发现二叉树本质上是一个个小二叉树组成的
在这里插入图片描述
那我们递归不就是把大事化小,复杂变简单吗?

因此我们就可以利用递归的思想进行实现

我们二叉树看成最简单的,也就下面几种情况
在这里插入图片描述
我们从根节点开始遍历,遇到空然后返回打印当前结点就好

🚩前序遍历

    // 前序遍历  根   左子树  右子树   递归
    public void preOrder(TreeNode root) {
        if(root == null) {
            return;
        }
        System.out.print(root.val+" ");
        preOrder(root.left);
        preOrder(root.right);
    }

🚩中序遍历

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

🚩后续遍历

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

    }

🛬前中后序练习题

  1. 某完全二叉树按层次输出(同一层从左到右)的序列为 ABCDEFGH 。该完全二叉树的前序序列为(A)
    A: ABDHECFG B: ABCDEFGH C: HDBEAFCG D: HDEBFGCA

  2. 二叉树的先序遍历和中序遍历如下:先序遍历:EFHIGJK;中序遍历:HFIEJKG.则二叉树根结点为(A)
    A: E B: F C: G D: H

  3. 设一课二叉树的中序遍历序列:badce,后序遍历序列:bdeca,则二叉树前序遍历序列为(D)
    A: adbce B: decab C: debac D: abcde

  4. 某二叉树的后序遍历序列与中序遍历序列相同,均为 ABCDEF ,则按层次输出(同一层从左到右)的序列为(A)
    A: FEDCBA B: CBAFED C: DEFCBA D: ABCDEF

总结:给出前序遍历与中序遍历和给出后序遍历与中序遍历可以确定一个二叉树,但是不给中序遍历或者只给一个中序遍历,是无法确定一个二叉树的

🐱‍🏍二叉树的基本操作

🎈获取树中节点的个数

依旧利用递归的思想,遍历每一棵小树,若当前结点为空,返回0

先获取左节点个数,再获取右节点个数

然后返回两者相加再加上根节点的个数1

比如以下结点:
在这里插入图片描述
若当前结点不为空,则返回1;

代码实现如下:

    public int size(BTNode root) {
        if (root == null) {
            return 0;
        }
        int leftSize = size(root.left);
        int rightSize = size(root.right);
        return leftSize + rightSize + 1;
    }

🎈获取叶子节点的个数

依旧利用递归的思想,遍历每一棵小树,若当前结点为空,返回0

当前节点的左右子树若都为空,说明该节点为叶子结点,返回1

先获取左节点个数,再获取右节点个数

然后两者相加

代码实现如下:

    int getLeafNodeCount(BTNode root) {
        if(root == null) {
            return 0;
        }
        if(root.left == null && root.right == null){
            return 1;
        }
        int leftSize = getLeafNodeCount(root.left);
        int rightSize = getLeafNodeCount(root.right);
        return leftSize+rightSize;
    }

🎈获取第K层节点的个数

依旧利用递归的思想,每进去一次,K-1,当k=1时,此时若该节点不为空则返回1

为空则返回0

先遍历左子树k层结点,再遍历右子树k层结点

最后左子树结点加上右子树结点,就是该层结点总数

    int getKLevelNodeCount(TreeNode root,int k) {
        if(root == null) {
            return 0;
        }
        if(k == 1) {
            return 1;
        }
        int leftSize = getKLevelNodeCount(root.left,k-1);
        int rightSize = getKLevelNodeCount(root.right,k-1);
        return leftSize+rightSize;
    }

🎈 获取二叉树的高度

分别统计左右子树的高度,然后进行比较

返回高度高的子树并加上根节点

    public int maxDepth(BTNode root) {
        if(root == null) {
            return 0;
        }
        int leftHeight = maxDepth(root.left);
        int rightHeight = maxDepth(root.right);

        return (leftHeight > rightHeight) ?
                (leftHeight+1):(rightHeight+1);
    }

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

依旧利用递归的思想

先遍历左子树,若没有找到,则返回null

若返回不为null,则返回该结点

若左子树没有,则遍历右子树,道理相同

若最后都没找到,则返回null;

    BTNode find(BTNode root, int val) {
        if (root == null) {
            return null;
        }
        if (root.val == val) {
            return root;
        }
        BTNode leftTree = find(root.left, val);
        if (leftTree != null) {
            return leftTree;
        }
        BTNode rightTree = find(root.right, val);
        if (rightTree != null) {
            return rightTree;
        }
        return null;//没有找到
    }

⭕总结

关于《【数据结构】二叉数的存储与基本操作的实现》就讲解到这儿,感谢大家的支持,欢迎各位留言交流以及批评指正,如果文章对您有帮助或者觉得作者写的还不错可以点一下关注,点赞,收藏支持一下!

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

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

相关文章

Typora mac版本安装

提示:文章介绍,Typora在Mac系统中免费安装使用 文章目录 一、官网下载二、安装 一、官网下载 官网地址:https://www.typoraio.cn/ 二、安装 安装好后按 command 空格键,找到 Typora的安装路径 /Applications/Typora.app/Con…

计网(第四章)(网络层)(六)

目录 一、路由选择协议(动态路由自动获取路由信息)概述: 二、因特网采用的路由协议 主要特点: 1.自适应 2.分布式 3.分层次 因特网采用分层次的路由选择协议: 三、常见的路由选择协议 一、路由选择协议&#xff…

遇到 Binder这些面试题,你会怎么答?

作为开发人员,每个人都有每个人擅长领域,自然也有自己不擅长的领域,很难成为完美的一个全栈开发。在面试中最怕遇见的一件事是面试官专挑你不擅长的领域进行提问,目的就是看你遇到问题的应变能力。 接下给大家分享一个面试中容易被…

使用 Netty 实现群聊功能的步骤和注意事项

文章目录 前言声明功能说明实现步骤WebSocket 服务启动Channel 初始化HTTP 请求处理HTTP 页面内容WebSocket 请求处理 效果展示总结 前言 通过之前的文章介绍,我们可以深刻认识到Netty在网络编程领域的卓越表现和强大实力。这篇文章将介绍如何利用 Netty 框架开发一…

报错sql_mode=only_full_group_by

首发博客地址 https://blog.zysicyj.top/ 报错内容 ### The error may exist in file[D:\code\cppCode20221025\leader-system\target\classes\mapper\system\TJsonDataMapper.xml] ### The error may involve defaultParameterMap ### The error occurred while…

Haproxy+Keepalive 整合rabbitmq实现高可用负载均衡

Haproxy 实现负载均衡 HAProxy 提供高可用性、负载均衡及基于 TCPHTTP 应用的代理,支持虚拟主机,它是免费、快速并且可靠的一种解决方案,包括 Twitter,Reddit,StackOverflow,GitHub 在内的多家知名互联网公司在使用。HAProxy 实现了一种…

康希诺的再估值:市场到底,行业向上

生物医药是整个二级市场弹性数一数二,但拐点难以揣摩的行业。这一点,美港A三大市场都曾经有过足够多的暴涨暴跌案例可用于佐证。 但很多时候,这种片面的表现又掩盖了生物医药自身的永续价值:在绝大多数细分赛道上,任何…

加密的PDF文件,如何解密?

PDF文件带有打开密码、限制编辑,这两种密码设置了之后如何解密? 不管是打开密码或者是限制编辑,在知道密码的情况下,解密PDF密码,我们只需要在PDF编辑器中打开文件 – 属性 – 安全,将权限状态修改为无保护…

基于金枪鱼群算法优化的BP神经网络(预测应用) - 附代码

基于金枪鱼群算法优化的BP神经网络(预测应用) - 附代码 文章目录 基于金枪鱼群算法优化的BP神经网络(预测应用) - 附代码1.数据介绍2.金枪鱼群优化BP神经网络2.1 BP神经网络参数设置2.2 金枪鱼群算法应用 4.测试结果:5…

链表(详解)

一、链表 1.1、什么是链表 1、链表是物理存储单元上非连续的、非顺序的存储结构,数据元素的逻辑顺序是通过链表的指针地址实现,有一系列结点(地址)组成,结点可动态的生成。 2、结点包括两个部分:&#x…

Fedora Linux 的家族(一):官方版本

导读本文将对 Fedora Linux 官方版本进行更详细的介绍。共有五个 版本: Fedora Workstation、Fedora Server、Fedora IoT、Fedora CoreOS 和 Fedora Silverblue。Fedora Linux 下载页面目前显示其中三个为 官方 版本,另外两个为 新兴 版本。本文将涵盖所…

js的this指向问题

代码一: 这段代码定义了run函数、obj对象,然后我们把run函数作为obj的方法。 function run(){console.log(this);}let obj{a:1,b:2};obj.runrun;obj.run(); 那么我们调用obj的run方法,那么这个方法打印的this指向obj。 分析:即…

【javaweb】学习日记Day4 - Maven 依赖管理 Web入门

目录 一、Maven入门 - 管理和构建java项目的工具 1、IDEA如何构建Maven项目 2、Maven 坐标 (1)定义 (2)主要组成 3、IDEA如何导入和删除项目 二、Maven - 依赖管理 1、依赖配置 2、依赖传递 (1)查…

11. 盛最多水的容器(c++题解)

11. 盛最多水的容器(c题解) 给定一个长度为 n 的整数数组 height 。有 n 条垂线,第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。 找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。 返回容器可以储存的最大…

分享一种针对uni-app相对通用的抓包方案

PART1,前言 近年来混合开发APP逐渐成为主流的开发模式,与传统的开发模式相比混合开发极大的提升了开发效率,同时跨平台的特性也降低了开发成本,一直以来混合开发被诟病的性能问题随着技术的发展也得到改善。技术的发展往往是一把…

HPC是如何助力AI推理加速的?

高性能计算(High-Performance Computing,HPC)通过提供强大的计算能力、存储资源和网络互联,可以显著地辅助人工智能(AI)应用更快地进行训练和推断。那么,HPC是如何助力AI推理加速的?…

多线程学习之线程池

线程状态 线程状态具体含义NEW一个尚未启动的线程的状态。也称之为初始、开始状态。线程刚被创建,但是并未启动。还没调用start方法。MyThread t new MyThread()只有线程对象,没有线程特征。RUNNABLE当我们调用线程对象的start方法,那么此时…

Java线程 - 详解(2)

一,线程安全问题 有些代码在单个线程的环境下运行,完全正确,但是同样的代码,让多个线程去执行,此时就可能出现BUG,这就是所谓的 "线程安全问题"。举一个例子: public class Demo {s…

python的可哈希对象

一、介绍 在Python中,可哈希(hashable)是指一种对象类型,该类型的对象可以用作字典的键(keys)或集合(sets)的元素。可哈希的对象具有以下特点: 不可变性(Imm…