二叉树及其遍历

news2024/11/26 3:53:17

文章目录

  • 二叉树
    • 树的定义
    • 二叉树的定义
    • 遍历
      • 先序遍历
      • 中序遍历
      • 后序遍历
      • 层次遍历
        • 定义队列
        • 层次创建二叉树
        • 层次遍历

二叉树

  树是一种非线性的数据结构,由若干个节点组成,节点之间存在一种父子关系,具有层次结构。二叉树是一种特殊的树结构,每个节点最多有两个子节点。树结构可以用来实现各种算法,例如二叉查找树、平衡二叉树、堆等。

树的定义

树(Tree)n(n>=0)个结点的有限集。n=0时称为空树。在任意一颗非空树中:

  1. 有且仅有一个特定的称为根(Root)的结点;
  2. 当n>1时,其余结点可分为m(m>0)个互不相交的有限集T1、T2、…、Tn,其中每一个集合本身又是一棵树,并且称为根的子树。

此外,树的定义还需要强调以下两点:

  1. n>0时根结点是唯一的,不可能存在多个根结点,数据结构中的树只能有一个根结点。
  2. m>0时,子树的个数没有限制,但它们一定是互不相交的。

二叉树的定义

  二叉树n(n>=0)个结点的有限集合,该集合或者为空集(称为空二叉树),或者由一个根结点和两棵互不相交的、分别称为根结点的左子树和右子树组成。

20131206141836-722390134
  从定义和图例中可以看出,二叉树每个节点最多只会有两棵子树,且左右子树是有顺序的,次序不能随意颠倒。当一个节点既没有左子树也没有右子树,则该节点为叶子节点

代码实现

结构体定义树

typedef struct Tree{
    int val; //数据域
    struct Tree *left; // 左子树
    struct Tree *right; // 右子树
}Tree,*tree;

遍历

  二叉树作为一种存储结构,将数据存入之后,则需要遍历对数据进行对应的处理。而二叉树的遍历分为四种:先(前)序遍历中序遍历后序遍历层次遍历

5c0fb4f0b1258

先序遍历

先序遍历是指从根节点开始,经过左子树,最后再遍历右子树,遍历顺序为:根节点->左子树->右子树。

代码实现

首先使用先序递归的创建一颗二叉树

// 创建新节点
Tree newNode(int val){
    Tree root = (Tree) malloc(sizeof (tree));
    root->val = val;
    root->left = NULL;
    root->right = NULL;
    return root
}
Tree CreateBiTree(int* len){//创建一颗节点数为len的二叉树
    if((*len)<=0){
        return NULL;
    }
    int val; //创建一个数据接收参数。
    printf("请输入插入数值:");
    // 为根节点数据域赋值
    scanf("%d",&val);
    //创建一个根节点
    Tree root = newNode(val);
    (*len)--;
    root->left = CreateBiTree(len); //递归创建左子树
    root->right = CreateBiTree(len); //递归创建右子树
    //创建完成后返回根节点
    return root;
}

再进行先序遍历

//先序遍历
void preorder(Tree root){
    if(root==NULL){
        return ;
    }
    // 首先输出根节点的值
    printf("节点的值:%d\n",root->val);
    //先递归遍历左子树
    preorder(root->left);
    //递归遍历右子树
    preorder(root->right);
}

运行结果

image-20230430105239154

中序遍历

中序遍历是指从左子树开始,再访问根节点,最后遍历右子树,遍历顺序为:左子树->根节点->右子树。

代码实现

利用先序递归创建一颗二叉树,并使用中序遍历二叉树

//中序遍历
void inorder(Tree root){
    //如果节点为NULL,退出遍历
    if(root==NULL){
        return ;
    }
    //先递归遍历左子树
    inorder(root->left);
    // 输出根节点的值
    printf("节点的值:%d\n",root->val);
    //递归遍历右子树
    inorder(root->right);
}

运行结果

image-20230430110915457

后序遍历

后序遍历是指从左子树开始,再遍历右子树,最后访问根节点,遍历顺序为:左子树->右子树->根节点。

代码实现

// 后序遍历
void postorder(Tree root){
    //如果节点为NULL,退出遍历
    if(root==NULL){
        return ;
    }
    //先递归遍历左子树
    postorder(root->left);
    //再递归遍历右子树
    postorder(root->right);
    //输出根节点的值
    printf("节点的值:%d\n",root->val);
}

运行结果

image-20230430111157150

  为什么后序遍历和中序遍历的结果相同呢?
  因为创建二叉树的时候使用的是前序递归,所以创建出来的二叉树都在根节点的左子树上,其右子树为空,此时这种情况被称为斜二叉树,并且也被称之为二叉树退化成单链表。这样创建出来的二叉树是很浪费空间且不规范的。
  所以先序递归创建的二叉树是不可取的。此时就用到层次创建二叉树,层次创建是用到最多的创建方式,也是较为直观的一种创建方式。

层次遍历

层次遍历是指从根节点开始,然后一层一层向下遍历。

代码实现

一把是利用队列来实现层次创建及遍历二叉树

定义队列

// 定义队列
struct Queue {
    struct Tree *Tree;
    struct Queue *next;
};

// 初始化队列
void initQueue(struct Queue **head, struct Queue **tail) {
    *head = *tail = NULL;
}

// 入队
void enQueue(struct Queue **head, struct Queue **tail, struct Tree *Tree) {
    struct Queue *node = (struct Queue*)malloc(sizeof(struct Queue));
    node->Tree = Tree;
    node->next = NULL;
    if (*head == NULL) {
        *head = *tail = node;
    } else {
        (*tail)->next = node;
        *tail = node;
    }
}

// 出队
struct Tree* deQueue(struct Queue **head, struct Queue **tail) {
    if (*head == NULL) {
        return NULL;
    }
    struct Tree *Tree = (*head)->Tree;
    if (*head == *tail) {
        *head = *tail = NULL;
    } else {
        *head = (*head)->next;
    }
    return Tree;
}

层次创建二叉树

// 创建二叉树
struct Tree* createTree(int *arr, int size) { //arr为数据数组,size为层数
    if (size == 0) {
        return NULL;
    }
    // 创建根节点
    struct Tree *root = (struct Tree*)malloc(sizeof(struct Tree));
    root->val = arr[0];
    root->left = NULL;
    root->right = NULL;
    // 初始化队列
    struct Queue *head, *tail;
    initQueue(&head, &tail);
    enQueue(&head, &tail, root);
    int i = 1;
    // 层次遍历创建二叉树
    while (i < size) {
        struct Tree *node = deQueue(&head, &tail);
        // 左子树
        if (i < size && arr[i] != -1) { //当数据为-1时证明该处为空节点
            node->left = (struct Tree*)malloc(sizeof(struct Tree));
            node->left->val = arr[i];
            node->left->left = NULL;
            node->left->right = NULL;
            enQueue(&head, &tail, node->left);
        }
        i++;
        // 右子树
        if (i < size && arr[i] != -1) {
            node->right = (struct Tree*)malloc(sizeof(struct Tree));
            node->right->val = arr[i];
            node->right->left = NULL;
            node->right->right = NULL;
            enQueue(&head, &tail, node->right);
        }
        i++;
    }
    return root;
}

层次遍历

// 层次遍历
void levelOrder(struct Tree* root) {
    if (root == NULL) {
        return;
    }
    struct Queue *head, *tail; // 定义队头与队尾
    initQueue(&head, &tail);
    enQueue(&head, &tail, root);
    while (head != NULL) {
        struct Tree *node = deQueue(&head, &tail);
        printf("%d ", node->val);
        if (node->left != NULL) {
            enQueue(&head, &tail, node->left);
        }
        if (node->right != NULL) {
            enQueue(&head, &tail, node->right);
        }
    }
}

main函数

// 测试代码
int main() {
    // 层次遍历序列,其中-1表示空节点
    int arr[] = {1, 2, 3, 4, -1, -1, 5};
    int size = sizeof(arr) / sizeof(int);
    // 创建二叉树
    struct Tree* root = createTree(arr, size);
    // 打印二叉树
    levelOrder(root);
    return 0;
}

运行结果

image-20230430162600792

层次遍历已经实现,接着使用层次创建二叉树,并实现先中后序遍历

运行结果分别为:

先序遍历

image-20230430163136783

中序遍历

image-20230430163210352

后序遍历

image-20230430163234047

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

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

相关文章

火遍全网的ChatGPT究竟是什么?

ChatGPT是什么 ChatGPT是一个由OpenAI开发的大型语言模型&#xff0c;基于GPT-3.5架构。它被训练用于自然语言处理和生成任务&#xff0c;可以回答各种问题&#xff0c;包括一般知识、文化、科学、技术、商业、娱乐等方面的问题。ChatGPT可以进行对话&#xff0c;回答用户的问…

OJ刷题 第十四篇(递归较多)

23204 - 进制转换 时间限制 : 1 秒 内存限制 : 128 MB 将一个10进制数x(1 < x < 100,000,000)转换成m进制数(2< m < 16) 。分别用 ABCDEF表示10以上的数字。 输入 x m (1 < x < 100,000,000, 2< m < 16) 输出 m进制数 样例 输入 31 16 输出 1F 答…

功能齐全的 ESP32 智能手表,具有多个表盘、心率传感器硬件设计

相关设计资料下载ESP32 智能手表带心率、指南针设计资料(包含Arduino源码+原理图+Gerber+3D文件).zip 介绍 我们调查了智能手表项目的不同方面,并学会了集成和测试每个单独的部分。在本文中,我们将使用所学知识,结合使用硬件和软件组件,从头开始创建我们自己的智能手表。在…

存储资源调优技术——SmartDedupe智能数据重删、SmartCompression智能数据压缩技术

目录 SmartDedupe智能数据重删技术 SmartCompression智能数据压缩技术 SmartDedupe智能数据重删技术 基本概念 智能数据重删技术 是一种数据缩减技术&#xff0c;通过删除存储系统中的冗余数据块 减少数据占用的物理存储容量&#xff0c;节省存储空间&#xff08;会降低性能&a…

Java 基础入门篇(三)——— 数组的定义与内存原理

文章目录 一、数组的定义1.1 静态初始化数组1.2 动态初始化数组1.3 数组的访问 二、数组的遍历三、数组的内存图 ★3.1 Java 的内存分配3.2 数组的内存图3.3 两个数组变量指向同一个数组对象 四、数组使用的常见问题补充&#xff1a;随机数 Random 类 一、数组的定义 数组就是…

线程池~~

文章目录 线程池线程池实现API、参数说明线程池处理Runnable任务线程池处理Callable任务Executors工具类实现线程池定时器Timer定时器ScheduledExecutorService定时器 并发和并行线程的生命周期 线程池 线程池实现API、参数说明 线程池处理Runnable任务 线程池处理Callable任务…

Win11的两个实用技巧系列之修改c盘大小方法、功能快捷键大全

Win11 c盘无法更改大小什么原因?Win11修改c盘大小方法 有不少朋友反应Win11 c盘无法更改大小是怎么回事&#xff1f;本文就为大家带来了详细的更改方法&#xff0c;需要的朋友一起看看吧 Win11 c卷无法更改大小什么原因&#xff1f;有用户电脑的系统盘空间太小了&#xff0c;…

CTF权威指南 笔记 -第二章二进制文件-汇编原理

C语言的生命是从 源文件开始 的 每条C语言都必须要给翻译成 一系列的低级语言 最后 按照可执行文件格式打包 并且作为二进制文件保存起来 编译原理 我们需要使用编译器 是通过某种语言 等价输出另一个语言 可以分为前端和后端 前端 和机器无关 把源程序分解为组成要素 …

【Android入门到项目实战-- 7.4】—— 如何播放音频和视频

目录 一、播放音频 MediaPlayer的工作流程 具体代码实现 二、播放视频 具体代码实现 学完本篇文章可以收获如何播放音频和视频。 一、播放音频 播放音频需要使用MediaPlayer类实现&#xff0c;它对各种格式的音频文件提供了全面的控制方法&#xff0c;下面是MediaPlayer类…

Android 9.0 原生SystemUI下拉通知栏UI背景设置为圆角背景的定制(二)

1.前言 在9.0的系统rom定制化开发中,在原生系统SystemUI下拉状态栏的通知栏的背景是默认白色四角的背景,由于在产品设计中,需要把四角背景默认改成圆角背景,所以就需要分析系统原生下拉通知栏的每条通知的默认背景,然后通过熟悉systemui的通知栏流程,设置默认下拉状态栏…

ChatGPT探索系列之三:探究ChatGPT的训练、优化和应用方法

文章目录 前言一、ChatGPT训练原理二、采样和微调阶段三、采样和训练奖励模型阶段三、采样和训练奖励模型阶段总结 前言 ChatGPT发展到目前&#xff0c;其实网上已经有大量资料了&#xff0c;博主做个收口&#xff0c;会出一个ChatGPT探索系列的文章&#xff0c;帮助大家深入了…

数据库如何合理生成主键:UUID、雪花算法

目录 1.使用自增主键的弊端 2.主键生成算法 2.1.UUID 2.1.1.概述 2.1.2.JAVA中的UUID 2.2.雪花算法 2.2.1.概述 2.2.2.JAVA中使用雪花算法 1.使用自增主键的弊端 首先在实际工程中我们很少用1&#xff0c;2&#xff0c;3......这样的自增主键&#xff0c;原因如下&…

MRI 成像方法和脉冲序列整理

以下内容为MRI期末复习笔记&#xff0c;仅供复习参考使用。 成像方法 2D重建 反投影重建 傅里叶重建 运用中心截面定理&#xff0c;一个三维物体的二维投影的傅里叶变换精确地等于物体的傅里叶变换的中心截面 上述定理换成二维同理 重建步骤 1、取得投影 P ( r , ϕ ) P(r,…

一招解决ChatGPT对话经常中断问题:KeepChatGPT插件

大家好&#xff0c;我是可夫小子&#xff0c;关注AIGC、读书和自媒体。解锁更多ChatGPT、AI绘画玩法。加&#xff1a;keeepdance&#xff0c;备注&#xff1a;chatgpt&#xff0c;拉你进群。 在访问ChatGPT官网进行聊天时&#xff0c;经常遇到了『与服务器断开』Something wet …

C++语法(22)---- 哈希表的闭散列和开散列

C语法&#xff08;21&#xff09;---- 模拟map和set_哈里沃克的博客-CSDN博客https://blog.csdn.net/m0_63488627/article/details/130354019?spm1001.2014.3001.5501 目录 1.哈希表的介绍 1.stl中的哈希表使用 2.比较 3.哈希的原理 4.哈希映射的方法 1.直接定址法 2.…

FreeRTOS 软件定时器

文章目录 一、软件定时器简介二、定时器服务/Daemon 任务三、单次定时器和周期定时器四、复位软件定时器1. 函数 xTimerReset()2. 函数 xTimerResetFromISR() 五、创建软件定时器1. 函数 xTiemrCreate()2. 函数 xTimerCreateStatic() 六、开启软件定时器1. 函数 xTimerStart()2…

条件构造器Wrapper

本文来说下mybatis-plus中的条件构造器Wrapper 文章目录 条件构造器程序实例 条件构造器 十分重要&#xff1a;Wrapper 我们写一些复杂的sql就可以使用它来替代&#xff01; 程序实例 测试一 Overridepublic List<OrderInfo> getOrderInfo() {// id等于2的数据QueryWrapp…

K210单片机的按键检测

这个图片是程序的效果&#xff0c;按下按键后蓝灯亮起&#xff0c;松开按键后蓝灯熄灭。 主要用的的就是函数的构造方法和使用方法&#xff1a; GPIO(ID,MODE,PULL,VALUE) GPIO 对象。 【ID】内部 GPIO 编号; 【MODE】GPIO 模式&#xff1b; GPIO.IN &#x…

保龄球游戏的获胜者、找出叠涂元素----2023/4/30

保龄球游戏的获胜者----2023/4/30 给你两个下标从 0 开始的整数数组 player1 和 player2 &#xff0c;分别表示玩家 1 和玩家 2 击中的瓶数。 保龄球比赛由 n 轮组成&#xff0c;每轮的瓶数恰好为 10 。 假设玩家在第 i 轮中击中 xi 个瓶子。玩家第 i 轮的价值为&#xff1a; …

SpringMVC学习总结(三)SpringMVC获取请求参数的几种方法解决获取请求参数的乱码问题CharacterEncodingFilter过滤器

SpringMVC学习总结&#xff08;三&#xff09;SpringMVC获取请求参数的几种方法/解决获取请求参数的乱码问题/CharacterEncodingFilter过滤器 一、通过ServletAPI获取请求参数 将HttpServletRequest作为控制器方法的形参&#xff0c;此时HttpServletRequest类型的参数表示封装…