【Java】二叉树:数据海洋中灯塔式结构探秘(上)

news2025/1/11 10:55:06

4289bfb309f34a118d9d0e3a4019831d.gif

个人主页 🌹:喜欢做梦


二叉树中有一个树,我们可以猜到他和树有关,那我们先了解一下什么是树,在来了解一下二叉树

一🍝、树型结构

1🍨.什么是树型结构?

树是一种非线性的数据结构,它是由n(n>=0)个有限节点(结点)和边组成的层次结构的集合。有一个特定的节点为根节点,其余节点通过边连接形成的分支,每个节点可以有零个或多个子节点。把它叫做树是因为它看起来像一颗倒挂的树,也就是说它是根朝上,而叶朝下的。

3d152da2704b40e58e8f702eebd81937.png655be38b9cc745ecb7505dace130970a.png

什么是线性结构?什么是非线性结构?

线性结构:数据元素呈现一对一的线性关系,除第一个和最后一个元素外,每个元素都有且仅有一个直接前驱和一个直接后继;

非线性结构:数据元素之间的关系不是简单的一对一,一个元素可能有多个前驱或后继,或者两者都有。

  • 树的定义是递归的
  • 除根节点外每一个结点都能引出一颗子树;
  • 树型结构中,子树之间不能有交集,否则就不是树型结构; 
  • 除了跟节点之外,每个节点有且只有一个父节点
  • 一个N个节点的树有N-1条边,因为根节点的上方没有边;

2🍩.什么是非树型结构?

非树:节点间的连通性复杂,可能存在多个路径连接统一对节点,也肯存在孤立节点,即与其他节点无连接。

002a5cd166de4deba327ac94ecc30971.png

3🍪.树型结构的基本性质

5e9a0482a612454cb7e8641f5af1dabd.png

  • 结点的度一个结点含有子树的个数称为该结点的度;如上图,A的度为3,C的度为2;
  • 树的度:一颗树中,所有结点度的最大值称为结点的度;如上图,树的度为4;
  • 叶子结点或终端结点度为0节点称为叶结点;如上图,E、F、G、P等结点为叶节点;
  • 孩子结点或子结点:一个结点含有子树的根结点称为该结点的子结点即只有根节点的结点才是子节点;如上图,B是A的孩子结点;
  • 双亲结点或父亲结点:若一个结点含有子结点,则这个树称为该结点的父结点;如图上A是B的父节点;
  • 根结点:一个树没有双亲的结点;如上图,A;
  • 结点的层次:从根开始定义,根为第一层,根的子结点为第二层,一次类推;
  • 树的高度或深度:树中结点的最大层次;如上图,树的高度为4;
  • 以下只需了解的概念:
  • 非终端结点或分支结点除根结点外,度不为0的结点;如上图:B、C、D、H为分支节点;
  • 兄弟结点:具有相同父结点的结点互称为兄弟结点;如上图:B、C是兄弟结点;
  • 堂兄弟结点不具有同一个父结点,但双亲在同一层的结点互为堂兄弟;如上图,G和H;
  • 结点的祖先:从根到该结点所经分支的所有结点;如上图,A就是所有结点的祖先;
  • 子孙:以某结点为根的子树中的任一节点,都称为该结点的子孙。如上图,所有结点都是A的子孙;
  • 森林:有m(m>=0)互不相交的树组成的集合称为森林;

4🍨.树的表现形式(了解)

树的表现形式有很多种,如双亲表示法,孩子表示法、孩子双亲表示法、孩子兄弟表示法等等。这里简单了解一下其中最常见的方法就是孩子兄弟表示法:

class Node{
    public int value;//数据
    public Node firstChild;//第一个孩子
    public Node nextBrother;//下一个兄弟
}

8a8938db550a44828dc848a6cbed7e39.png

二🍝、二叉树

1🍑.什么是二叉树?

二叉树:二叉树是每个结点最多有两科子树的树的结构,其两个子树通常称为左子树和右子树

二叉树的递归定义:

  • 或者是一颗空树;
  • 或者是一颗由一根结点和两课互不相交的分别称为左子树和右子树所组成的非空数,左子树和右子树又同样是二叉树;

0948a72b839d46c0ab3b3a38cbab082f.png

特点:

  • 度的限制:结点的度最大为2;
  • 有序性:左右子树由顺序,即使某节点只有一颗子树,也要区分左右子树;

性质:

  • 若规定的根节点层数为1,这一棵非空二叉树的第i层上最多有eq?2%5E%7Bi-1%7D(i>.0)个节点;
  • 若规定只有根节点的二叉树的深度为1,则深度为k的二叉树的最大节点数是eq?2%5E%7Bk%7D-1(k>=0);
  • 对于任何一棵二叉树,如果其叶节点个数为n0,度为2的非叶节点个数为n2,则有n0=n2+1;
  • 对于一颗有n个节点的m个二叉树(m表示树的度),其最小深度计算公式:eq?%5Clog_%7Bm%7D(n(m-1)+1);

f12a02b2492a4d8890a3a466d63462d1.png

2🍑.二叉树的类型

1.满二叉树

满二叉树:每一层的结点树都达到最大,除最后一层外每个节点都有两个节点。

eeeabfbbc6c24894aab44b7d586a191d.png

特点:

  • 节点度数:除最后一层的叶子节点外,其他层的每一个的节点都有两个节点,即度都为2;
  • 叶子节点:所有的叶子节点都在同一层,且叶子节点的数量为eq?2%5E%7Bk-1%7D,k为数的高度;
  • 节点总数: 节点总数是eq?2%5E%7Bk%7D-1

9563a474a20f42bdb433782d220cb2fd.png

2.完全二叉树

完全二叉树:除最后一层外,其余层节点数都达到最大,最后一层节点从左到右依次按顺序排列,可通过数组的高效和访问,完全二叉树是满二叉树的一种

d85c274e594b47ed9cf6f4426afbd706.png

特点:

  • 节点度数:除了底层的叶子节点外,其余所有节点都有两个子节点,即度数均为2;
  • 叶子节点分布:所有叶子节点都在同一层,这使得树的结构呈现出完美的形态;
  • 具有n个节点的完全二叉树的深度k为eq?%5Clog_%7B2%7D(n+1)上取整进1;

总节点个数与叶子节点个数的关系: 

  • 如果总节点个数为偶数,那么叶子节点为总节点个数的一半,也就是n=2n0;
  • 如果总节点个数为奇数,那么叶子节点为总节点个数的一半再减一,也就是n=2n0-1;
  • 节点数量:对于具有n个节点的完全二叉树,如果按照从上至下从左至右的顺序对所有节点从0开始编号,这对于序号为i的节点有:
  • 若i>0,双亲序号:(i-1)/2;i=0,i为根节点的编号,无双亲节点;
  • 若2i+1<n,左孩子序号:2i+1,否则无左孩子;
  • 若2i+1<n,右孩子序号:2i+2,否则无右孩子; 

49929b888a0d4fb0b9029af25eea53d2.png

3🍪.二叉树的创建

fde4be67a312411f8ce3e8200c0cbc40.png

public class BinaryTree {
    public static class TreeNode{
          public char val;//数据
          public TreeNode left;//左孩子
          public TreeNode right;//右孩子
        public TreeNode(char val) {
            this.val = val;
        }
    }
    public TreeNode createTree(){
        //创建节点
        TreeNode A=new TreeNode('A');
        TreeNode B=new TreeNode('B');
        TreeNode C=new TreeNode('C');
        TreeNode D=new TreeNode('D');
        TreeNode E=new TreeNode('E');
        TreeNode F=new TreeNode('F');
        TreeNode G=new TreeNode('G');
        //连接节点
        A.left=B;
        A.right=C;
        B.left=D;
        B.right=E;
        C.left=F;
        C.right=G;
        return A;
    }
}

4.二叉树的遍历

二叉树的遍历是指按照一定的顺序访问二叉树中的每个节点,且每个节点仅被访问一次

二叉树的遍历方式主要有前序遍历、中序遍历、后序遍历;

前序遍历

前序遍历:遍历顺序是先访问根的的节点—>左子树—>右子树,也就是根、左、右;

a198f3fdace44be695330449675b29c1.png

 前序遍历代码:

// 前序遍历
    public void preOrder(TreeNode root){
        //判断是否有节点,没有返回
        if(root == null){
            return;
        }
        System.out.print(root.val+ " ");
        //遍历左子树
        preOrder(root.left);
        //遍历右子树
        preOrder(root.right);
    }

a0b4e7ef1c374aba8cf29baefc46b1ff.png

3f8e67209be0487fbd5c413401238c44.png bcc53479fa0f476497bf02c8bbeefb6f.png

  •  顺序:根节点--左子树--右子树;
  • 根结点的打印位置:第一个;

 中序遍历

中序遍历:遍历顺序是先访问左子树—>根的的节点—>右子树,也就是左、根、右;

54ccf1923033436dbdc0fa94ff200515.png

中序遍历代码: 

 // 中序遍历
    public void inOrder(TreeNode root){
        //判断是否有节点,没有返回
        if(root == null){
            return;
        }
        //遍历左子树
        preOrder(root.left);
        System.out.print(root.val+ " ");
        //遍历右子树
        preOrder(root.right);
    }

 

818481e07b5a4139aa2738f04d458d17.png 49562fc8df8443389426a81adc9cb705.png

b0d17ae796024aed99e4a3d4bab0663d.png d5360f9190eb498f8eebf9dca12fc3ce.png

 

0d0e081c78684a21af4330dbc76905f9.png

  • 顺序:左子树--根节点--右子树;
  • 根结点的打印位置:中间;

 后序遍历

 后序遍历:遍历顺序是先访问左子树—>右子树—>根的的节点,也就是左、根、右;

8d9a82cfe4554255ab36bc2c48051013.png

    // 后序遍历
    public void postOrder(TreeNode root){
        //判断是否有节点,没有返回
        if(root == null){
            return;
        }
        //遍历左子树
        preOrder(root.left);
        //遍历右子树
        preOrder(root.right);
        System.out.print(root.val+ " ");
    }
}

后序遍历的过程与前面的也是同理,就不画图过多解释了。 

  • 顺序:左子树--右子树--根节点;
  • 根结点的打印位置:最后一个;

三者之间的区别:

 前序遍历中序遍历后序遍历
访问顺序根、左、右左、根、右左、右、根
根节点访问位置第一个中间最后一个
应用场景二叉树结构、将表达式树转换为前缀表达式用于输出有序序列,还能辅助将表达式树转换为中缀表达式二叉树的高度、节点数,以及释放二叉树内存

 

层序遍历

层序遍历:从上至下,从左至右逐层访问就是层序遍历。

63a1afe44ad840ef8f7306852070d820.png

层序遍历的代码,我后期补上,或者下篇在写,这篇就到这里啦~ 

961e1dc9cbf04a488e6a5d9204a7355c.jpeg

 

我的博客即将同步至腾讯云开发者社区,邀请大家一同入驻:https://cloud.tencent.com/developer/support-plan?invite_code=3k88uddoizs48

 

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

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

相关文章

在 macOS 上安装 MongoDB Community Edition

https://www.mongodb.com/zh-cn/docs/manual/tutorial/install-mongodb-on-os-x/

docker部署单机版doris

文章目录 前言一、系统环境简介二、部署要求三、部署安装1、基础设置2、下载镜像3、下载安装包4、启动镜像环境5、配置fe6、配置be 总结 前言 应项目测试需求&#xff0c;需使用docker部署单机版doris。 一、系统环境简介 #1 系统信息 [roottest][~] $cat /etc/redhat-relea…

c++编程玩转物联网:使用芯片控制8个LED实现流水灯技术分享

在嵌入式系统中&#xff0c;有限的GPIO引脚往往限制了硬件扩展能力。74HC595N芯片是一种常用的移位寄存器&#xff0c;通过串行输入和并行输出扩展GPIO数量。本项目利用树莓派Pico开发板与74HC595N芯片&#xff0c;驱动8个LED实现流水灯效果。本文详细解析项目硬件连接、代码实…

uni-app运行 安卓模拟器 MuMu模拟器

最近公司开发移动端系统&#xff0c;使用真机时每次调试的时候换来换去的麻烦&#xff0c;所以使用模拟器来调试方便。记录一下安装和连接的过程 一、安装MuMu模拟器 百度搜索MuMu模拟器并打开官网或者点这里MuMu模拟器官网 点击下载模拟器 安装模拟器&#xff0c;如果系统…

C语言解析命令行参数

原文地址&#xff1a;C语言解析命令行参数 – 无敌牛 欢迎参观我的个人博客&#xff1a;无敌牛 – 技术/著作/典籍/分享等 C语言有一个 getopt 函数&#xff0c;可以对命令行进行解析&#xff0c;下面给出一个示例&#xff0c;用的时候可以直接copy过去修改&#xff0c;很方便…

传奇996_36——背包图标,物品位置问题

绑定位置不对位 CTRLF9背包物品文件&#xff0c;也就是bag_item文件夹的bag_item.lua文件&#xff0c;这个小框和大框的相对位置会影响那个绑定图标,就是背包物品组合的标签和下面子标签的相对位置 背包物品偏移到看不见 原因&#xff1a;CTRLF9背包物品文件&#xff0c;也就…

使用Python和Pybind11调用C++程序(CMake编译)

目录 一、前言二、安装 pybind11三、编写C示例代码四、结合Pybind11和CMake编译C工程五、Python调用动态库六、参考 一、前言 跨语言调用能对不同计算机语言进行互补&#xff0c;本博客主要介绍如何实现Python调用C语言编写的函数。 实验环境&#xff1a; Linux gnuPython3.10…

如何选择黑白相机和彩色相机

我们在选择成像解决方案时黑白相机很容易被忽略&#xff0c;因为许多新相机提供鲜艳的颜色&#xff0c;鲜明的对比度和改进的弱光性能。然而&#xff0c;有许多应用&#xff0c;选择黑白相机将是更好的选择&#xff0c;因为他们产生更清晰的图像&#xff0c;更好的分辨率&#…

代码美学:MATLAB制作渐变色

输入颜色个数n&#xff0c;颜色类型&#xff1a; n 2; % 输入颜色个数 colors {[1, 0, 0], [0, 0, 1]}; createGradientHeatmap(n, colors); 调用函数&#xff1a; function createGradientHeatmap(n, colors)% 输入检查if length(colors) ~ nerror(输入的颜色数量与n不一…

SAP 零售方案 CAR 系统的介绍与研究

前言 当今时代&#xff0c;零售业务是充满活力和活力的业务领域之一。每天&#xff0c;由于销售运营和客户行为&#xff0c;它都会生成大量数据。因此&#xff0c;公司迫切需要管理数据并从中检索见解。它将帮助公司朝着正确的方向发展他们的业务。 这就是为什么公司用来处理…

【leetcode】动态规划

31. 873. 最长的斐波那契子序列的长度 题目&#xff1a; 如果序列 X_1, X_2, ..., X_n 满足下列条件&#xff0c;就说它是 斐波那契式 的&#xff1a; n > 3对于所有 i 2 < n&#xff0c;都有 X_i X_{i1} X_{i2} 给定一个严格递增的正整数数组形成序列 arr &#xff0…

自动泊车“哐哐撞大墙”,小米SU7智驾功能bug缠身?

文/王俣祺 导语&#xff1a;小米SU7&#xff0c;自带热度与科技光环的“流量神车”&#xff0c;近日却以一种极为“狼狈”的方式闯入大众视野。多达70余辆小米SU7陷入“泊车魔咒”&#xff0c;瞬间在网络上炸开了锅。从“科技控”到“惹祸精”的背后&#xff0c;究竟藏着怎样的…

优维HAO案例:全球TOP15汽车零件供应商「IT运维自动化」创新工程

撰文&#xff1a;鹿小U / 制图&#xff1a;脾气超好 又是一家很厉害的客户。 YADT是全(hu)球(zhēn)领(ji)先(sh)的汽车座椅供应商&#xff0c;拥有从汽车座椅零部件到整椅的完整生产制造能力&#xff0c;为中国几乎所有的汽车制造商提供汽车整椅产品和服务。 YADT在国内拥…

Linux宝塔部署wordpress网站更换服务器IP后无法访问管理后台和打开网站页面显示错乱

一、背景&#xff1a; wordpress网站搬家&#xff0c;更换服务器IP后&#xff0c;如果没有域名时&#xff0c;使用服务器IP地址无法访问管理后台和打开网站页面显示错乱。 二、解决方法如下&#xff1a; 1.wordpress搬家后&#xff0c;在新服务器上&#xff0c;新建站点时&am…

MyBatis框架-动态SQL-XML中的常用标签+特殊字符在XML中的显示

一、if标签、where标签、trim标签、choose标签、set标签、foreach标签 1、问题引入&#xff1a;where关键字和and关键字在动态SQL里面应该如何添加&#xff1f; &#xff08;1&#xff09;if标签&#xff1a; test属性的值是判断条件 if标签里面的内容是条件成立时添加到SQ…

EXTI配置流程 含中断延时消抖点亮小灯

如图可知&#xff0c;配置流程分成以下一个部分 ①使能GPIO时钟 __HAL_RCC_GPIOA_CLK_ENABLE();// 打开时钟 ②初始化利用 HAL_GPIO_Init 一步到位&#xff0c;可以初始化外设GPIO的一切 4个参数 &#xff08;引脚 Pull 这里选择的模式是从下面这几个里面选 速度&#x…

数据库系列之GaussDB数据库高可用能力测试验证

数据库的高可用能力是数据库的基本能力&#xff0c;可靠性的设计和机制能够保证数据库节点异常时能够正常切换、减少业务的影响范围和时间&#xff0c;保证业务的可用性和连续性。本文主要介绍GaussDB数据库的高可用能力测试验证情况&#xff0c;通过模拟故障场景来验证GaussDB…

Android BottomNavigationView 底部导航栏使用详解

一、BottomNavigationView简介 BottomNavigationView是官方提供可以实现底部导航的组件&#xff0c;最多支持5个item&#xff0c;主要用于功能模块间的切换&#xff0c;默认会包含动画效果。 官方介绍地址&#xff1a;BottomNavigationView 二、使用BottomNavigationView a…

IIT开发自适应协作界面,通过Xsens动作捕捉系统实现安全人机交互

意大利理工学院(IIT)的研究人员正在利用自适应界面转变人机协作&#xff0c;实现实时机器人调整和安全、无缝的交互。 本文要点: l 协作的实时适应&#xff1a;IIT的研究员西林图纳和厄兹达马尔开发了一种自适应协作界面(ACI)&#xff0c;允许机器人根据人类的运动意图实时调…

一学就废|Python基础碎片,格式化F-string

Python 3.6 中引入了 f-string语法&#xff0c;提供了一种简洁直观的方法来将表达式和变量直接嵌入到字符串中进行字符串格式化&#xff0c;f -string背后的想法是使字符串插值更简单。 要创建 f -string&#xff0c;在字符串前加上字母 “f”即可&#xff0c;与字符串本身的格…