初识二叉树和二叉树的基本操作

news2025/1/17 1:44:34

目录

一、树 

 1.什么是树

2. 与树相关的概念

二、二叉树 

1.什么是二叉树

2.二叉树特点

3.满二叉树与完全二叉树

4.二叉树性质

 相关题目:

5.二叉树的存储 

6.二叉树的遍历和基本操作 

 二叉树的遍历

二叉树的基本操作


一、树 

 1.什么是树

  • 子树是不相交的;
  • 除了根结点外,每个结点有且仅有一个父结点
  • 一棵N个结点的树有N-1条边。 

树:                                                    非树: 

       

2. 与树相关的概念

  • 结点的度:一个结点含有子树的个数称为该结点的度; 如上图:A的度为6
  • 树的度:一棵树中,所有结点度的最大值称为树的度; 如上图:树的度为6
  • 叶子结点或终端结点:度为0的结点称为叶结点; 如上图:B、C、H、I...等节点为叶结点
  • 双亲结点或父结点:若一个结点含有子结点,则这个结点称为其子结点的父结点; 如上图:A是B的父结点
  • 孩子结点或子结点:一个结点含有的子树的根结点称为该结点的子结点; 如上图:B是A的孩子结点
  • 根结点:一棵树中,没有双亲结点的结点;如上图:A
  • 结点的层次:从根开始定义起,根为第1层,根的子结点为第2层,以此类推
  • 树的高度或深度:树中结点的最大层次; 如上图:树的高度为4

二、二叉树 

1.什么是二叉树

一棵二叉树是结点的一个有限集合,该集合:

  • 或者为空
  • 或者是由一个根节点加上两棵别称为左子树右子树的二叉树组成。

 

2.二叉树特点

  • 二叉树不存在度大于2的结点(树的度:一棵树中,所有结点 度的最大值 称为树的度)
  • 二叉树的子树有左右之分,次序不能颠倒,因此二叉树是有序树 

3.满二叉树与完全二叉树

  • 满二叉树: 一棵二叉树,如果每层的结点数都达到最大值,则这棵二叉树就是满二叉树。也就是说,如果一棵又树的层数为K,且结点总数是2^k-1,则它就是满二叉树

  • 完全二叉树: 完全二叉树是效率很高的数据结构,完全二叉树是由满二叉树而引出来的。对于深度为K的,有n个结点的二叉树,当且仅当其每一个结点都与深度为K的满二叉树中编号从0至n-1的结点一 一对应时称之为完全二叉树。 意思就是完全二叉树的所有节点从上到下,从左到右依次排满。 要注意的是满二叉树是一种特殊的完全二又树。 

完全二叉树有一个特点,那就是如果总结点数为奇数,那么这个二叉树就只有一个度为1的节点,如果是偶数,就没有度为1的结点。 

4.二叉树性质

1. 若规定根结点的层数为1,则一棵非空二叉树的第i层上最多有 2^{i}-1(i>0)个结点

2. 若规定只有根结点的二叉树的深度为1,则深度为K的二叉树的最大结点数是2^{k}-1(k>=0)

3. 对任何一棵二叉树, 如果其叶结点个数为 n0, 度为2的非叶结点个数为 n2,则有n0=n2+1

4. 具有n个结点的完全二叉树的深度k为\log _{_{2}}(n+1)上取整

5. 对于具有n个结点的完全二叉树,如果按照从上至下从左至右的顺序对所有节点从0开始编号,则对于序号为i 的结点有:

  • 若i>0,双亲序号:(i-1)/2;i=0,i为根结点编号,无双亲结点
  • 若2i+1<n,左孩子序号:2i+1,否则无左孩子
  • 若2i+2<n,右孩子序号:2i+2,否则无右孩子 

 相关题目:

1. 某二叉树共有 399 个结点,其中有 199 个度为 2 的结点,则该二叉树中的叶子结点数为( )
A 不存在这样的二叉树
B 200
C 198
D 199
2.在具有 2n 个结点的完全二叉树中,叶子结点个数为( )
A n
B n+1
C n-1
D n/2
3.一个具有767个节点的完全二叉树,其叶子节点个数为()
A 383
B 384
C 385
D 386
4.一棵完全二叉树的节点数为531个,那么这棵树的高度为( )
A 11
B 10
C 8
D 12 

答案:
1.B

解析:

叶子结点或终端结点:度为0的结点称为叶结点。
由前面说的二叉树性质第3点:对任何一棵二叉树, 如果其叶结点个数为 n0, 度为2的非叶结点个数为 n2,则有n0=n2+1。

所以 n2=199,n0=n2+1=200.

2.A

解析: 

设结点总数为N=2n,因为题目中说了这是一颗完全二叉树,而结点总数是偶数,那么说明这个二叉树只有一个度为1的结点。
N=n0+n1+n2 => 2n=n0+n2+1 因为n0=n2+1,所以  2n-1=n0+n0-1 => n0=n

3.B 

解析: 

N=767,767为奇数,所以这个二叉树没有度为1的结点(n1=0)
N=n0+n1+n2=n0+n0-1=767 => n0=384

4.B 

解析: 

由前面说的二叉树性质第4点:具有n个结点的完全二叉树的深度k为l\log _{_{2}}(n+1)上取整.

2^{9}< 532 <2^{10}  => 9<\log _{_{2}}532 <10,因为是上取整,那么 k=10.

5.二叉树的存储 

二叉树的存储结构分为:
顺序存储类似于链表的链式存储。二叉树的链式存储是通过一个一个的节点引用起来的,常见的表示方式有二叉和三叉表示方式(孩子表示法以及孩子双亲表示法)。 

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

6.二叉树的遍历和基本操作 

二叉树的遍历

1.前中后序遍历

  • 前序遍历:根节点 左子树 右子树
  • 中序遍历:左子树 根节点 右子树
  • 后序遍历:左子树 右子树 根节点

2.层序遍历

自上而下,自左至右逐层访问树的结点的过程就是层序遍历

有如下二叉树,大家可用上述方法自行遍历 

前序遍历:A B D E H C F G

中序遍历:D B E H A F C G 

后序遍历:B D E H C F G A 

层序遍历:A B C D E F G H 

代码实现: 

这里先按照上图用穷举的方式快速构建一颗二叉树(不是构建二叉树的正确方法) 

public class BinaryTree {
    public static class TreeNode {
        TreeNode left;
        TreeNode right;
        char val;

        TreeNode(char val) {
            this.val = val;
        }
    }

    private TreeNode root;

    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');
        TreeNode H = new TreeNode('H');

        root = A;
        A.left = B;
        A.right = C;

        B.left = D;
        B.right = E;
        C.left = F;
        C.right = G;

        E.right = H;
        return A;
    }

    //前序遍历
    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;
        }
        preOrder(root.left);
        preOrder(root.right);
        System.out.print(root.val+" ");
    }

}
public class Main {
    public static void main(String[] args) {
        BinaryTree binaryTree=new BinaryTree();
        BinaryTree.TreeNode root=binaryTree.createTree();

        System.out.print("前序遍历:");
        binaryTree.preOrder(root);
        System.out.println();
        System.out.print("中序遍历:");
        binaryTree.inOrder(root);
        System.out.println();
        System.out.print("后序遍历:");
        binaryTree.postOrder(root);
    }
}

运行结果:

二叉树的基本操作

1. 获取树中节点的个数

这个方法实现在这里有两种思路:

1.遍历这个树,是结就nodeSize++
2.用子问题的思路来解决:总结点数=左子树结点的个数+右子树结点的个数+根结点

    public static int nodeSize=0;
    //获取树中节点的个数(遍历每个节点)
    public void size(TreeNode root){
        if (root==null){
            return;
        }
        nodeSize++;
        size(root.left);
        size(root.right);
    }
    //用子问题的思路来解决:总节点数=左子树节点的个数+右子树节点的个数+根节点
    public int size2(TreeNode root){
        if (root==null){
            return 0;
        }

        int tmp=size2(root.left)+size2(root.right)+1;
        return tmp;
    }

2.获取叶子节点的个数

叶子结点的特点就是度为0,即其左子树和右子树都是空。

这个方法实现在这里有两种思路:

1.遍历这个树,只要root不为空且root的左子树和右子树都为空,就说明root所在的结点是叶子结点
2.用子问题的思路来解决:总叶子结点数=左子树的叶子结点+右子树的叶子结点

    public int leafSize;
    public void getLeafNodeCount(TreeNode root) {
        if(root == null) {
            return;
        }
        if(root.left == null && root.right == null) {
            leafSize++;
        }
        getLeafNodeCount(root.left);
        getLeafNodeCount(root.right);
    }

    //子问题思路:这颗树的总叶子结点数=左子树的叶子结点+右子树的叶子结点
    public int getLeafNodeCount2(TreeNode root) {
        if (root == null) {
            return 0;
        }
        if (root.left == null && root.right == null) {
            return 1;
        }
        return getLeafNodeCount2(root.left)
                + getLeafNodeCount2(root.right);
    }

3.获取第K层节点的个数

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

4.获取二叉树的高度

 整棵树的高度=找出 左子树的高度 和 右子树的高度 的最大值 +1(树的高度或深度:树中结点的最大层次)

    // 获取二叉树的高度
    public int getHeight(TreeNode root){
        if(root==null){
            return 0;
        }

        int leftHeight=getHeight(root.left);
        int rightHeight=getHeight(root.right);

        return Math.max(leftHeight,rightHeight)+1;
    }

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

1.先判断根节点的值是不是我们要找的value,如果是就返回这个root

2.如果当前根节点不是我们要找的value,那就到当前根节点的左子树去找,如果左子树找不到就去右子树找。

// 检测值为value的元素是否存在
    private TreeNode find(TreeNode root, int val){
        if (root==null){
            return null;
        }
        if (root.val==val){
            System.out.println(root.val);
            return root;
        }

        TreeNode leftval=find(root.left,val);
        if(leftval!=null){
            return leftval;
        }

        TreeNode rightval=find(root.right,val);
        if (rightval!=null){
            return rightval;
        }

        return null;
    }

6.层序遍历

先入队根节点,然后出队,若当前根节点左右不为空,则把不为空的左右入队,出新的队头,以此类推。  

   public void levelOrder(TreeNode root) {
        if(root == null) {
            return;
        }
        Queue<TreeNode> queue = new LinkedList<>();
        queue.offer(root);
        while (!queue.isEmpty()) {
            TreeNode cur = queue.poll();
            System.out.print(cur.val+" ");
            if(cur.left != null) {
                queue.offer(cur.left);
            }
            if(cur.right != null) {
                queue.offer(cur.right);
            }
        }
    }

7.判断一棵树是不是完全二叉树

1.先把根节点放到队列当中

2.队列不为空,弹出元素,带入左右(可以为空)

3.当队列弹出元素为null则停止

4.最后一步,判断当前队列是否元素都是nul,只要出现不为nul的元素,则当前二又树不是完全二叉树 

   public boolean isCompleteTree(TreeNode root) {
        if(root == null) return true;
        Queue<TreeNode> queue = new LinkedList<>();
        queue.offer(root);
        while (!queue.isEmpty()) {
            TreeNode cur = queue.poll();
            if(cur != null) {
                queue.offer(cur.left);
                queue.offer(cur.right);
            }else {
                break;//结束之后  遍历队列剩下的所有元素 是不是都是null
            }
        }
        // 遍历队列剩下的所有元素 是不是都是null
        while (!queue.isEmpty()) {
            TreeNode cur = queue.poll();
            if(cur != null) {
                return false;
            }
        }
        return true;
    }

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

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

相关文章

windows server 2019-DHCP服务器搭建

一、DHCP的原理&#xff1a; 1&#xff0c;获得ip的过程 客户机发送请求给所有DHCP服务器 DHCP服务器收到请求后发送响应包给客户机 客户机收到响应包&#xff08;第一个到达的&#xff09;后&#xff0c;检验响应包里面的ip能不能用&#xff08;使用ping的方式&#xff0c…

Ideal的使用技巧

一、springcloud项目如何将多个服务放到services中一起启动 1、打开ideal&#xff0c;再view -> Tool Windows -> services 2、在services界面 找到 run configuration type -> springboot即可 二、配置临时的启动参数 1、在edit configurations中 2、选择相应的服务…

网络原理 - HTTP / HTTPS(5)——https协议

目录 一、HTTPS是什么 为什么要进行加密 二、“加密” 是什么 三、HTTPS的工作过程 &#xff08;1&#xff09;引入对称加密 对称密钥的特点&#xff1a; &#xff08;2&#xff09;引入非对称加密 非对称加密的特点&#xff1a; &#xff08;3&#xff09;中间人攻击…

深入浅出 -- 系统架构之负载均衡Nginx反向代理

一、Nginx反向代理-负载均衡 首先通过SpringBootFreemarker快速搭建一个WEB项目&#xff1a;springboot-web-nginx&#xff0c;然后在该项目中&#xff0c;创建一个IndexNginxController.java文件&#xff0c;逻辑如下&#xff1a; Controller public class IndexNginxControl…

免费SSL证书申请指南

在互联网时代&#xff0c;HTTPS安全协议已成网站标配&#xff0c;而SSL/TLS证书是实现HTTPS的关键。以下是如何申请免费证书的简明流程&#xff1a; 1.选择证书颁发机构&#xff08;CA&#xff09;&#xff1a;现今很多知名CA如JoySSL、Lets Encrypt等提供免费SSL证书服务。选定…

云备份day02

&#x1f4df;作者主页&#xff1a;慢热的陕西人 &#x1f334;专栏链接&#xff1a;C云备份项目 &#x1f4e3;欢迎各位大佬&#x1f44d;点赞&#x1f525;关注&#x1f693;收藏&#xff0c;&#x1f349;留言 主要内容介绍了第三方库jsoncpp和bundle库的使用 文章目录 云备…

关于Linux下的进程状态(进程篇)

目录 Linux操作系统的一般进程状态 关于阻塞 关于挂起 Linux内核状态源代码&#xff1a; 关于僵尸进程 关于孤儿进程 Linux操作系统的一般进程状态 新建&#xff1a;字面意思运行&#xff1a;task_struct结构体在运行队列中排队&#xff0c;就叫做运行态阻塞&#xff1a;等待非…

Java项目:基于Springboot+vue社区医院管理系统设计与实现(源码+数据库+毕业论文)

一、项目简介 本项目是一套基于Springbootvue社区医院管理系统 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;eclipse或者idea 确保可以运行&#xff01; 该系统功能完善、界面美观、操作简单、功能…

机器学习知识点全面总结

机器学习按照模型类型分为监督学习模型、无监督学习模型两大类。 1、有监督学习 有监督学习通常是利用带有专家标注的标签的训练数据&#xff0c;学习一个从输入变量X到输入变量Y的函数映射。Y f (X)&#xff0c;训练数据通常是(nx,y)的形式&#xff0c;其中n代表训练样本的大…

基于springboot实现甘肃非物质文化网站系统项目【项目源码+论文说明】

摘要 现代经济快节奏发展以及不断完善升级的信息化技术&#xff0c;让传统数据信息的管理升级为软件存储&#xff0c;归纳&#xff0c;集中处理数据信息的管理方式。本甘肃非物质文化网站就是在这样的大环境下诞生&#xff0c;其可以帮助管理者在短时间内处理完毕庞大的数据信…

Java Number类(详细)

一&#xff0c;Number java.lang.Number 类是抽象类&#xff0c;用于表示各种数值类型的对象。它是 Java 中所有数值类型的包装类的超类。Number 类主要有以下几个特点和作用&#xff1a; 抽象类&#xff1a;Number 类是一个抽象类&#xff0c;不能直接实例化&#xff0c;但可以…

计算机中数的表示

0. 简介 介绍计算机中数的表示方法&#xff0c;主要内容来自 c s a p p csapp csapp。 1. 整数的表示 包括有符号整数与无符号整数的表示。 假设 w → [ w n − 1 w n − 2 . . . w 0 ] \overrightarrow w[w_{n-1}w_{n-2}...w_0] w [wn−1​wn−2​...w0​] 为一种整数。…

【Python系列】将生成的 JSON 数据写入 JSON 文件

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

嵌入式开发学习---(部分)数据结构(无代码)

数据结构 为什么学习数据结构&#xff1f; 1&#xff09;c语言告诉如何写程序&#xff0c;数据结构是如何简洁高效的写程序 2&#xff09;遇到一个实际问题&#xff0c;需要写程序去实现相应功能&#xff0c;需要解决那两个方面的问题&#xff1f; 如何表达数据之间的逻辑规律…

智慧公厕,城市管理新亮点

在现代城市环境建设中&#xff0c;智慧公厕作为智慧环卫的重要组成部分&#xff0c;正以其先进的技术手段&#xff0c;成为城市管理的亮点和标杆。借助物联网等技术的应用&#xff0c;智慧公厕实现了信息化、数字化、智慧化的全面升级&#xff0c;为市民提供了更便捷舒适的公共…

[WIP]Sora相关工作汇总VQGAN、MAGVIT、VideoPoet

视觉任务相对语言任务种类较多(detection, grounding, etc.)、粒度不同 (object-level, patch-level, pixel-level, etc.)&#xff0c;且部分任务差异较大&#xff0c;利用Tokenizer核心则为如何把其他模态映射到language space&#xff0c;并能让语言模型更好理解不同的视觉任…

Linux安装及管理程序(rpm)

目录 一.Linux应用程序基础 1.应用程序与系统命令的关系 2.典型应用程序的目录结构 3.常见的软件包封装类型 二.RPM包管理工具 1.rmp 查询 2.rpm安装与卸载 3.维护 RPM 数据库 一.Linux应用程序基础 1.应用程序与系统命令的关系 对比系统命令和应用程序的不同 1.1位…

聚簇索引与非聚簇索引b+树实现的区别

文章目录 聚簇索引非聚簇索引B树中聚簇索引的查找&#xff08;匹配&#xff09;逻辑B树中非聚簇索引的查找&#xff08;匹配&#xff09;逻辑 聚簇索引 特点&#xff1a; 索引和数据保存在同一个B树中 页内的记录是按照主键的大小顺序排成一个单向链表 。 页和页之间也是根据…

Vite 项目中环境变量的配置和使用

Vite 项目中环境变量的声明 我们要在 Vite 项目中进行环境变量的声明&#xff0c;那么需要在项目的根目录下&#xff0c;新建 .env.[mode] 文件用于声明环境变量&#xff0c;如&#xff1a; .env.test 文件用于测试环境下项目全局变量的声明.env.dev 文件用于开发环境下项目全…

FastWiki发布`0.2.4`支持js 函数

Release v0.2.4 AIDotNet/fast-wiki (github.com) 支持JS动态functioncall调用支持动态function管理支持JS在线编辑提供智能代码提示支持JS在线编辑提供部分绑定的c#类&#xff08;默认提供Console&#xff0c;HttpClient&#xff09;支持Application绑定多个Function Call优…