【数据结构】二叉树的链式结构

2023-10-27

【数据结构】二叉树的链式存储结构

二叉树的存储结构

typedef int BTDataType;
// 二叉树的结构
typedef struct BinaryTreeNode {
    BTDataType data;             // 树的值
    struct BinaryTreeNode *left; // 左孩子
    struct BinaryTreeNode *right;// 右孩子
} BinaryTreeNode;

二叉树的深度优先遍历

在这里插入图片描述

前序遍历

前序遍历,又叫先根遍历。
遍历顺序:根 -> 左子树 -> 右子树

代码:

// 前序遍历
void BinaryTreePrevOrder(BinaryTreeNode *root) {
    // 前序遍历 根、左子树、右子树
    // 如果root为空,递归结束
    if (root == NULL) {
        printf("NULL ");
        return;
    }

    printf("%d ", root->data);
    BinaryTreePrevOrder(root->left);
    BinaryTreePrevOrder(root->right);
}

中序遍历

中序遍历,又叫中根遍历。
遍历顺序:左子树 -> 根 -> 右子树

代码

// 中序遍历
void BinaryTreeInOrder(BinaryTreeNode *root) {
    // 中序遍历 - 左子树、根、右子树
    if (root == NULL) {
        printf("NULL ");
        return;
    }
    BinaryTreeInOrder(root->left);
    printf("%d ", root->data);
    BinaryTreeInOrder(root->right);
}

后序遍历

后序遍历,又叫后根遍历。
遍历顺序:左子树 -> 右子树 -> 根

代码

// 后序遍历
void BinaryPostOrder(BinaryTreeNode *root) {
    // 后序遍历 - 左子树、右子树、根
    if (root == NULL) {
        printf("NULL ");
        return;
    }
    BinaryPostOrder(root->left);
    BinaryPostOrder(root->right);
    printf("%d ", root->data);
}

二叉树的广度优先遍历

层序遍历

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

在这里插入图片描述

思路(借助一个队列):

  1. 先把根入队列,然后开始从队头出数据。
  2. 出队头的数据,把它的左孩子和右孩子依次从队尾入队列(NULL不入队列)。
  3. 重复进行步骤2,直到队列为空为止。

img

特点:借助队列先进先出的性质,上一层数据出队列的时候带入下一层数据。

// 层序遍历 - 利用队列
void BinaryTreeLevelOrder(BinaryTreeNode *root) {
    // 创建一个队列,队列中入数据,取队头,然后带左右子树,出一个数,带一个树的所有子树
    Queue q;
    QueueInit(&q);
    if (root) {
        // root不为NULL,就入队列
        QueuePush(&q, root);
    }

    while (!QueueEmpty(&q)) {
        // 取队头,打印
        BinaryTreeNode *front = QueueFront(&q);
        printf("%d ", front->data);

        // 取完POP
        QueuePop(&q);
        // 取队头,带下一层,
        if (front->left) {
            QueuePush(&q, front->left);
        }

        if (front->right) {
            QueuePush(&q, front->right);
        }
    }
    printf("\n");
    QueueDestroy(&q);
}

二叉树的节点个数

求解树的节点总数时,可以将问题拆解成子问题:

  1. 若为空,则结点个数为0。
  2. 若不为空,则结点个数 = 左子树节点个数 + 右子树节点个数 + 1(自己)。

代码

// 求二叉树节点个数
int BinaryTreeSize(BinaryTreeNode *root) {
    if (root == NULL) {
        return 0;
    }
    return 1 + BinaryTreeSize(root->left) + BinaryTreeSize(root->right);
}

二叉树的叶子节点个数

子问题拆解:

  1. 若为空,则叶子结点个数为0。
  2. 若结点的左指针和右指针均为空,则叶子结点个数为1。
  3. 除上述两种情况外,说明该树存在子树,其叶子结点个数 = 左子树的叶子结点个数 + 右子树的叶子结点个数。

代码:

// 求二叉树的叶子节点个数
int BinaryTreeLeafSize(BinaryTreeNode *root) {
    if (root == NULL) {
        return 0;
    }

    if (root->left == NULL && root->right == NULL) {
        return 1;
    }

    return BinaryTreeLeafSize(root->left) + BinaryTreeLeafSize(root->right);
}

第K层节点的个数

思路:
相对于根结点的第k层结点的个数 = 相对于以其左孩子为根的第k-1层结点的个数 + 相对于以其右孩子为根的第k-1层结点的个数。

在这里插入图片描述

代码

// 求第k层的节点个数 k>=1
int BinaryTreeLevelSize(BinaryTreeNode *root, int k) {
    if (root == NULL) {
        return 0;
    }

    if (k == 1) {
        return 1;
    }

    return BinaryTreeLevelSize(root->left, k - 1) + BinaryTreeLevelSize(root->right, k - 1);
}

值为x的节点

子问题:

  1. 先判断根结点是否是目标结点。
  2. 再去左子树中寻找。
  3. 最后去右子树中寻找。

代码

// 二叉树查找值为x的节点
BinaryTreeNode *BinaryTreeFind(BinaryTreeNode *root, BTDataType x) {
    if (root == NULL) {
        return NULL;
    }

    if (root->data == x) {
        return root;
    }

    // 左子树递归的节点保存到leftTree中,如果leftTree不为NULL,则return leftTree
    BinaryTreeNode *leftTree = BinaryTreeFind(root->left, x);
    if (leftTree) {
        return leftTree;
    }

    // 右子树递归的节点保存到rightTree中,如果rightTree不为NULL,则return rightTree
    BinaryTreeNode *rightTree = BinaryTreeFind(root->right, x);
    if (rightTree) {
        return rightTree;
    }

    // 找不到,返回NULL
    return NULL;
}

树的高度

子问题:

  1. 若为空,则深度为0。
  2. 若不为空,则树的最大深度 = 左右子树中深度较大的值 + 1。

代码

// 求二叉树的高度
int BinaryTreeHeight(BinaryTreeNode *root) {
    // 求左子树的高度,右子树的高度
    // 取最大的
    if (root == NULL) {
        return 0;
    }
    int leftHeight = BinaryTreeHeight(root->left);
    int rightHeight = BinaryTreeHeight(root->right);
	
	//如果左右子树两边相等就取左边的高度,所以大于等于
    return leftHeight >= rightHeight ? leftHeight + 1 : rightHeight + 1;
}

判断二叉树是否是完全二叉树

判断二叉树是否是完全二叉树的方法与二叉树的层序遍历类似,但又有一些不同。

思路(借助一个队列):

  1. 先把根入队列,然后开始从队头出数据。
  2. 出队头的数据,把它的左孩子和右孩子依次从队尾入队列(NULL也入队列)。
  3. 重复进行步骤2,直到读取到的队头数据为NULL时停止入队列。
  4. 检查队列中剩余数据,若全为NULL,则是完全二叉树;若其中有一个非空的数据,则不是完全二叉树。

在这里插入图片描述

代码

// 判断二叉树是否是完全二叉树
bool BinaryTreeComplete(BinaryTreeNode *root) {
    // 层序走,走到第一个为空的时候,就跳出去,如果是满二叉树,后面的节点都应该为空
    Queue q;
    QueueInit(&q);
    if (root) {
        QueuePush(&q, root);
    }

    while (!QueueEmpty(&q)) {
        BinaryTreeNode *front = QueueFront(&q);
        QueuePop(&q);

        if (front == NULL) {
            break;
        } else {
            QueuePush(&q, front->left);
            QueuePush(&q, front->right);
        }
    }
    // 出到空以后,如果后面全是空,则是完全二叉树
    while (!QueueEmpty(&q)) {
        BinaryTreeNode *front = QueueFront(&q);
        QueuePop(&q);
        if (front != NULL) {
            QueueDestroy(&q);
            return false;
        }
    }
    QueueDestroy(&q);
    return true;
}

判断二叉树是否是单值二叉树

单值二叉树,所有节点的值都相同的二叉树即为单值二叉树,判断某一棵二叉树是否是单值二叉树的一般步骤如下:

  1. 判断根的左孩子的值与根结点是否相同。
  2. 判断根的右孩子的值与根结点是否相同。
  3. 判断以根的左孩子为根的二叉树是否是单值二叉树。
  4. 判断以根的右孩子为根的二叉树是否是单值二叉树。

若满足以上情况,则是单值二叉树。

注:空树也是单值二叉树。

代码

//求单值二叉树
bool isUnivalTree(BinaryTreeNode *root) {
    if (root == nullptr) {
        return true;
    }

    if (root->left && root->data != root->left->data) {
        return false;
    }
    
	if (root->right && root->data != root->right->data) {
        return false;
    }

    return isUnivalTree(root->left) && isUnivalTree(root->right);
}

判断二叉树是否是对称二叉树

对称二叉树,这里所说的对称是指镜像对称:

要判断某二叉树是否是对称二叉树,则判断其根结点的左子树和右子树是否是镜像对称即可。因为是镜像对称,所以左子树的遍历方式和右子树的遍历方式是不同的,准确来说,左子树和右子树的遍历是反方向进行的。

代码

//求对称二叉树
bool _isSymmetric(BinaryTreeNode *left, BinaryTreeNode *right) {
    // 两个都为NULL,对称
    if (left == NULL && right == NULL)
        return true;

    // 两个其中一个为NULL,一个不为NULL,不对称
    if (left == NULL || right == NULL)
        return false;

    // left的左孩子的值和right的值不相等,不对称
    if (left->data != right->data)
        return false;

    // 左子树的左孩子,和右子树的右孩子对比,然后左子树的右孩子和右子树的左孩子在对比
    return _isSymmetric(left->left, right->right) && _isSymmetric(left->right, right->left);
}

bool isSymmetric(BinaryTreeNode *root) {
    if (root == nullptr) {
        return true;
    }

    return _isSymmetric(root->left, root->right);
}

翻转二叉树

思路:

  1. 翻转左子树。
  2. 翻转右子树。
  3. 交换左右子树的位置。

代码

BinaryTreeNode *invertTree(BinaryTreeNode *root) {
    if (root == nullptr) {
        return nullptr;
    }

    BinaryTreeNode *leftTree = invertTree(root->left);
    BinaryTreeNode *rightTree = invertTree(root->right);

    root->left = rightTree;
    root->right = leftTree;

    return root;
}

二叉树的构建和销毁

// 申请树节点
BinaryTreeNode *BuyBinaryTreeNode(BTDataType x) {
    BinaryTreeNode *newnode = (BinaryTreeNode *) malloc(sizeof(BinaryTreeNode));
    if (newnode == NULL) {
        perror("malloc fail");
        exit(-1);
    }
    newnode->data = x;
    newnode->left = newnode->right = NULL;
    return newnode;
}

// 通过前序遍历的数组"ABD##E#H##CF##G##"构建二叉树
BinaryTreeNode *BinaryTreeCreate(BTDataType *a, int *pi) {
    if (a[*pi] == '#') {
        (*pi)++;
        return NULL;
    }

    BinaryTreeNode *root = (BinaryTreeNode *) malloc(sizeof(BinaryTreeNode));
    if (root == NULL) {
        perror("malloc fail");
        exit(-1);
    }
    root->data = a[(*pi)++];

    root->left = BinaryTreeCreate(a, pi);
    root->right = BinaryTreeCreate(a, pi);
    return root;
}

销毁

// 二叉树销毁
void BinaryTreeDestory(BinaryTreeNode **root) {
    if (*root == NULL) {
        return;
    }
    // 后序遍历销毁,根要最后释放
    BinaryTreeDestory(&(*root)->left);
    BinaryTreeDestory(&(*root)->right);
    free(*root);
    *root = NULL;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【数据结构】二叉树的链式结构 的相关文章

随机推荐

  • Windows server 2016——权限管理与数据恢复

    作者简介 一名云计算网络运维人员 每天分享网络与运维的技术与干货 公众号 网络豆 座右铭 低头赶路 敬事如仪 个人主页 网络豆的主页 目录 写在前面 一 SQL server 的安全机制 1 设置 SQL server 权限 2 登录权限设
  • C++(入门基础)缺省参数、函数重载、引用、内联函数

    文章目录 一 命名空间 命名空间定义 命名空间使用 二 缺省参数 备胎 全缺省参数 半缺省参数 三 函数重载 四 引用 引用权限的放大和缩小 引用的特性 常引用 引用的使用 引用和指针的区别 五 内联函数 内联的特性 宏的优缺点 c 有哪些
  • CUDA 入门教程

    CUDA从入门到精通 零 写在前面 在老板的要求下 本博主从2012年上高性能计算课程开始接触CUDA编程 随后将该技术应用到了实际项目中 使处理程序加速超过1K 可见基于图形显示器的并行计算对于追求速度的应用来说无疑是一个理想的选择 还有
  • Redhat7.6中Oracle19c单机版安装

    最近在学习Oracle19c的安装中 借鉴博客 从中遇到了问题并整理 避免踩坑 环境前的配置及oracle应用的安装 转载 redhat7 6Linux安装Oracle19C完整版教程 tomhaha 博客园 在配置数据库实例的时候 转载
  • React合成事件(阻止冒泡stopImmediatePropagation)

    文章目录 一 遇到的问题 问题描述 分析 二 React 合成事件 1 执行顺序 2 合成事件阻止冒泡 2 1e stopPropagation 2 2 e nativeEvent stopImmediatePropagation 3 th
  • java 整合 Elastic 8.

    1 准备工作 使用docker 快速搭建的环境 官网docker compose 方式搭建的集群 设置了密码登录 elastic elastic 需要给jdk 导入证书 找到 证书对应目录 复制到桌面 主要导入下面2个证书 执行如下命令 k
  • 输入学生学号、姓名、三科成绩计算出平均分保存至指定文件中

    include
  • baichuan-53B VS ChatGLM-6B对比

    由于百川智能的内测模型是baichuan 53B 尽管模型大小不一致 为了方便 我们仍然选择百川智能baichuan 53B与ChatGLM 6B内测结果进行对比 其中ChatGLM 6B的结果来自https github com THUD
  • 常见编程错误

    编码示例 内存相关 内存相关 1 scanf d val 读一个整数到一个变量 正确应当传递变量地址 2 bss内存位置 诸如未初始化的全局X变量 总是被加载器初始化为0 但是对于堆内存却并不是这样的 需要程序员显示地将分配的堆内存初始化
  • 【智能时代的颠覆】AI让物联网不再是物联网

    自我介绍 我是秋说 研究人工智能 大数据等前沿技术 传递Java Python等语言知识 主页链接 秋说的博客 学习专栏推荐 MySQL进阶之路 C 刷题集 网络安全攻防姿势总结 欢迎点赞 收藏 留言 如有错误敬请指正 引言 人工智能 AI
  • iscsi删除已失效的链路

    有套rac环境 主机连接存储使用的iscsi方式 使用了一段时间 客户感觉网络设计不合理 需要调整网段vlan和ip地址 首先关闭实例和集群 调整存储端和主机端的ip地址和vlan 调整后可以ping通 使用如下命令配置 两台主机都需要配置
  • MyBatis自动生成实体类(逆向工程)

    mybatis自动生成代码工具 逆向工程 MyBatis自动生成实体类 逆向工程 MyBatis属于一种半自动的ORM框架 它需要我们自己编写sql语句和映射文件 但是编写映射文件和sql语句很容易出错 所以mybatis官方提供了Gene
  • 计算机视觉之三维重建(三)(单视图测量)

    2D变换 等距变换 旋转平移 保留形状 面积 通常描述刚性物体运动 相似变换 在等距变换的基础增加缩放特点 射影变换 共线性 四共线点的交比保持不变 仿射变换 面积比值 平行关系等不变 仿射变换是特殊的射影变换 影消点与影消线 2D无穷远点
  • 《C陷阱与缺陷》学习笔记

    C编译器判断符号的方式是 贪心法 即一直读入下一字符 看能否组成一个符号 直到不可能组成一个符号为止 单引号括起的一个字符表示一个整数 双引号括起的一个字符代表一个指针 float g g是一个函数 该函数的返回值类型为指向浮点数的指针 f
  • 日本核污水排海:普通民众的个人防护指南

    面对日本核污水排海的问题 普通民众需要采取一些个人防护措施 以确保自身的健康与安全 本文将提供一些实用的指南 帮助普通民众做好个人防护 减少潜在的风险 一 了解核污水排放的情况 首先 我们需要充分了解关于核污水排放的背景 科学依据以及相关的
  • SpringMVC:整合JQUERY与JSON

    原文地址 http liuzidong iteye com blog 1069343 参考资料 1 Spring3 MVC 笔记 二 json rest优化 http 7454103 iteye com show full true 2 j
  • 本翻译专栏的说明

    我是一名计算机专业在校学生 主攻C 我英语水平一般 请大家轻喷 我会利用课余时间来翻译cplusplus网站中我感兴趣的内容 最后 祝大家看得开心 有所收获 2023年3月27日制定的翻译计划 Reference的C library的
  • 第二篇 溢出标志 CF与OF

    在汇编学习中 个人感觉CF与OF这两个溢出标志还是有点难理解的 笔者也还是一知半解 若有错误之处 请指正 一 学习CF与OF 要始终牢记一点 CF是无符号数溢出标志 OF是有符号数溢出标志 通俗一点说就是 即使有符号数相加 相减导致了CF
  • STM32(HAL库)——光电编码器、M/T法测量电机转速

    目录 一 编码器 二 电机测试的三种方法 三 STM32CubeMx配置 四 程序编写 五 实验结果 一 编码器 常见的用于电机测速的编码器有霍尔编码器和光电编码器两种 两者测速的基本原理不同 但都是输出两路相位差90 的脉冲信号 这里以光
  • 【数据结构】二叉树的链式结构

    数据结构 二叉树的链式存储结构 二叉树的存储结构 typedef int BTDataType 二叉树的结构 typedef struct BinaryTreeNode BTDataType data 树的值 struct BinaryTr