当前位置: 首页 > article >正文

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

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.kler.cn/a/410640.html

相关文章:

  • JVM-类文件结构
  • 填补覆盖空白,小型机器人让智能清洁再“净”一步!
  • qt添加模块
  • 替换Nacos的MySQL驱动
  • 线性代数的发展简史
  • 一个关于 CSS Modules 的陷阱
  • Hive的基础函数-日期函数
  • 【从经典到最新】28种信号分解方法,EMD类,VMD类,小波类,FMD、多元分解等(附matlab代码)
  • HDMI转VGA方案 LT8612UX(HDMI2.0) LT8612SX LT8511EX LT8522EX LT8612EX_e(HDMI1.4)
  • 企业数字化转型现状
  • 学习日志016--python实现双向循环列表与链栈
  • 软件测试丨Python语法与数据结构
  • C++【面试重要题目】 只出现一次的数字的集合.
  • git推送报错443
  • 从零开始:NetBox 4.1 Docker 部署和升级
  • 嵌入式的C/C++:深入理解 static、const 与 volatile 的用法与特点
  • 4.3 使用 JMeter 发起请求详解
  • 【人工智能】基于PyTorch的深度强化学习入门:从DQN到PPO的实现与解析
  • python VS c++
  • 室内定位论文速递(11.18-11.22)
  • Visual Studio下载安装教程(非常详细)从零基础入门到精通,看完这一篇就够了_visual studio安装教程
  • 鸿蒙征文|鸿蒙心路旅程:从零到一的探索与成长——我的HarmonyOS
  • 如何定制谷歌浏览器的外观主题
  • 基于IPMI的服务器硬件监控指标解读
  • CSS笔记(一)炉石传说卡牌设计1
  • 周志华深度森林deep forest(deep-forest)最新可安装教程,仅需在pycharm中完成,超简单安装教程