【数据结构】二叉树链式结构详解

目录

  • 1.前言
  • 2.快速创建一颗二叉树
  • 3.二叉树的遍历
    • 3.1前序遍历
    • 3.2中序遍历
    • 3.3后序遍历
    • 3.4层序遍历
  • 4.二叉树节点个数与高度
    • 4.1二叉树节点个数
    • 4.2二叉树叶子节点个数
    • 4.3二叉树高度
    • 4.4二叉树第k层节点个数
    • 4.5二叉树查找值为x的节点
  • 5.二叉树的基础oj题练习
  • 6.二叉树的创建和销毁
    • 6.1通过前序遍历的数组"ABD##E#H##CF##G##"构建二叉树
    • 6.2二叉树销毁
    • 6.3判断二叉树是否是完全二叉树

1.前言

前面我们已经讲过二叉树的概念及堆的实现,而我们所学的堆其实只是二叉树的顺序结构实现,当然堆的实现只能适用于完全二叉树或者满二叉树,但如果不是完全二叉树或者满二叉树这样的结构呢?那么就必须要谈一谈二叉树的链式结构的实现了,接下来我们将对递归有着更深层次的去理解,准备好小本本吧!

2.快速创建一颗二叉树

在学习二叉树的基本操作前,需要先创建一棵二叉树,然后才能学习其相关的基本操作。
由于现在大家对二叉树结构掌握还不够深入,为了降低学习成本,此处手动快速创建一棵简单的二叉树,快速进入二叉树操作学习,等二叉树结构了解的差不多时,我们反过头再来研究二叉树真正的创建方式。
下面看代码:

typedef int BTDataType;
typedef struct BinaryTreeNode
{BTDataType data;struct BinaryTreeNode* left;struct BinaryTreeNode* right;
}TreeNode;TreeNode* BuyTreeNode(int x)
{TreeNode* node = (TreeNode*)malloc(sizeof(TreeNode));assert(node);node->data = x;node->left = NULL;node->right = NULL;return node;
}TreeNode* CreateTree()
{TreeNode* node1 = BuyTreeNode(1);TreeNode* node2 = BuyTreeNode(2);TreeNode* node3 = BuyTreeNode(3);TreeNode* node4 = BuyTreeNode(4);TreeNode* node5 = BuyTreeNode(5);TreeNode* node6 = BuyTreeNode(6);TreeNode* node7 = BuyTreeNode(7);node1->left = node2;node1->right = node4;node2->left = node3;node4->left = node5;node4->right = node6;node5->right = node7;return node1;
}

注意:上述代码并不是创建二叉树的方式,真正创建二叉树方式后序详解重点讲解。

在开始二叉树的基本操作前,先回顾一下我们二叉树的概念,二叉树:

  1. 空树
  2. 非空:跟节点,根节点的左子树,根节点的右子树组成的。
    在这里插入图片描述
    所以说二叉树可以更好的理解递归,二叉树的概念就说明二叉树是递归式的,二叉树的基本操作都是递归式的。
    上面我们创建的二叉树加了一个新节点,便于后面的理解。
    在这里插入图片描述

3.二叉树的遍历

学习二叉树结构,最简单的方式就是遍历。所谓二叉树遍历(Traversal)是按照某种特定的规则,依次对二叉树中的节点进行相应的操作,并且每个节点只操作一次。访问结点所做的操作依赖于具体的应用问题。
遍历是二叉树上最重要的运算之一,也是二叉树上进行其它运算的基础。

按照规则,二叉树的遍历有:前序/中序/后序的递归结构遍历:

  1. 前序遍历(Preorder Traversal 亦称先序遍历)——访问根结点的操作发生在遍历其左右子树之前。
  2. 中序遍历(Inorder Traversal)——访问根结点的操作发生在遍历其左右子树之中(间)。
  3. 后序遍历(Postorder Traversal)——访问根结点的操作发生在遍历其左右子树之后。

由于被访问的结点必是某子树的根,所以N(Node)、L(Left subtree)和R(Right subtree)又可解释为根、根的左子树和根的右子树。NLR、LNR和LRN分别又称为先根遍历、中根遍历和后根遍历。

忘了更便于我们理解,我们把空也打印出来,下面就是这几种遍历方式。

3.1前序遍历

前序遍历就是
分治思想:根 左子树 右子树
根据这个思想进行遍历得到前序序列。
代码实现:

void PrevOrder(TreeNode* root)
{if (root == NULL){printf("N ");return;}printf("%d ", root->data);PrevOrder(root->left);PrevOrder(root->right);
}

前序遍历结果:1 2 3 N N N 4 5 N 7 N N 6 N N

3.2中序遍历

中序遍历就是
分治思想: 左子树 根 右子树
根据这个思想进行遍历得到中序序列。
代码实现:

void InOrder(TreeNode* root)
{if (root == NULL){printf("N ");return;}InOrder(root->left);printf("%d ", root->data);InOrder(root->right);
}

中序遍历结果:N 3 N 2 N 1 N 5 N 7 N 4 N 6 N

3.3后序遍历

后序遍历就是
分治思想:左子树 右子树 根
根据这个思想进行遍历得到后序序列。
代码实现:

void BackOrder(TreeNode* root)
{if (root == NULL){printf("N ");return;}BackOrder(root->left);BackOrder(root->right);printf("%d ", root->data);

后序遍历结果:N N 3 N 2 N N N 7 5 N N 6 4 1

3.4层序遍历

层序遍历有两个版本,版本一可是直接遍历一遍二叉树并打印一行,
而版本二可以分层进行打印,一层一层的打印出数据。
对于层序遍历中的队列的使用,可以去搬来我之前写的栈和队列那一节内容的代码。
版本一:

void BinaryTreeLevelOrder(TreeNode* root)
{Queue q;QueueInit(&q);if (root)QueuePush(&q,root);while (!QueueEmpty(&q)){TreeNode* front = QueueFront(&q);QueuePop(&q);printf("%d ", front->data);if (front->left)QueuePush(&q,front->left);if (front->right)QueuePush(&q,front->right);}QueueDestroy(&q);
}

版本二:

void BinaryTreeLevelOrder(TreeNode* root)
{Queue q;QueueInit(&q);if (root)QueuePush(&q, root);int levelSize = 1;while (!QueueEmpty(&q)){while (levelSize--){TreeNode* front = QueueFront(&q);QueuePop(&q);printf("%d ", front->data);if (front->left)QueuePush(&q, front->left);if (front->right)QueuePush(&q, front->right);}printf("\n");levelSize = QueueSize(&q);}QueueDestroy(&q);
}

4.二叉树节点个数与高度

4.1二叉树节点个数

int TreeSize(TreeNode* root)
{return root == NULL ? 0 : TreeSize(root->left) + TreeSize(root->right) + 1;
}

4.2二叉树叶子节点个数

int BinaryTreeLeaf(TreeNode* root)
{//空树 返回0if (root == NULL)return 0;//左子树为空,右子树为空 是叶子,返回1if (!root->left && !root->right)return 1;//递归遍历,不是空也不是叶子,分治左右子树之和return BinaryTreeLeaf(root->left) + BinaryTreeLeaf(root->right);
}

4.3二叉树高度

int TreeHeight(TreeNode* root)
{if (root == NULL)return 0;return fmax(TreeHeight(root->left), TreeHeight(root->right)) + 1;
}

一道Leetcode的一道题。
求二叉树的深度OJ链接
解题代码:

int maxDepth(struct TreeNode* root) {if (root == NULL)return 0;int LeftHeight = maxDepth(root->left);int RightHeight = maxDepth(root->right);return LeftHeight > RightHeight ? LeftHeight + 1 : RightHeight + 1;
}

4.4二叉树第k层节点个数

int TreeLevelK(TreeNode* root, int k)
{assert(k > 0);if (root == NULL)return 0;if (k == 1)return 1;return TreeLevelK(root->left, k - 1) + TreeLevelK(root->right, k - 1);
}

4.5二叉树查找值为x的节点

BTNode* BinaryTreeFind(BTNode* root, BTDataType x)
{if (root == NULL)return NULL;if (root->data == x){return root;}BTNode* ret1 = BinaryTreeFind(root->left, x);if (ret1)return ret1;BTNode* ret2 = BinaryTreeFind(root->right, x);if (ret2)return ret2;return NULL;
}

5.二叉树的基础oj题练习

  1. 单值二叉树。OJ链接
bool isUnivalTree(struct TreeNode* root) {if(root == NULL)return true;if(root->left && root->val != root->left->val)return false;if(root->right && root->val != root->right->val)return false;return isUnivalTree(root->left) && isUnivalTree(root->right);
}
  1. 检查两颗树是否相同。OJ链接
bool isSameTree(struct TreeNode* p, struct TreeNode* q) {//均为空if(p == NULL && q == NULL)return true;//有一个为空if(p == NULL || q == NULL)return false;//都不为空且不相等if(p->val != q->val)return false;return isSameTree(p->left,q->left)&& isSameTree(p->right,q->right);
}
  1. 对称二叉树。OJ链接
 bool _isSymmetric(struct TreeNode* root1, struct TreeNode* root2) {//都为空if(root1 == NULL && root2 == NULL){return true;}//其中一个为空if(root1 == NULL || root2 == NULL){return false;}//都不为空if(root1->val != root2->val){return false;}return _isSymmetric(root1->left,root2->right)&& _isSymmetric(root1->right,root2->left);
}
bool isSymmetric(struct TreeNode* root) {return _isSymmetric(root->left, root->right);
}
  1. 二叉树的前序遍历。 OJ链接
 int TreeSize(struct TreeNode* root){return root == NULL ? 0 : TreeSize(root->left) + TreeSize(root->right) + 1;}
int* pre_order(struct TreeNode* root, int* a, int* pi)
{if(root == NULL)return NULL;a[(*pi)++] = root->val;pre_order(root->left,a,pi);pre_order(root->right,a,pi);return a;
}
int* preorderTraversal(struct TreeNode* root, int* returnSize) {int n = TreeSize(root);int* a = (int*)malloc(sizeof(int)*n);*returnSize = n;int i = 0;return pre_order(root, a,&i);
}
  1. 二叉树中序遍历 。OJ链接
 int TreeSize(struct TreeNode* root){return root == NULL ? 0 : TreeSize(root->left) + TreeSize(root->right) + 1;}int* inorder(struct TreeNode* root, int* s, int* pi){if(root == NULL)return NULL;inorder(root->left,s,pi);s[(*pi)++] = root->val;inorder(root->right,s,pi);return s;}
int* inorderTraversal(struct TreeNode* root, int* returnSize) {int n = TreeSize(root);int* s = (int*)malloc(sizeof(int)*n);*returnSize = n;int i = 0;return inorder(root,s,&i);
}
  1. 二叉树的后序遍历 。OJ链接
int TreeSize(struct TreeNode* root){return root == NULL ? 0 : TreeSize(root->left) + TreeSize(root->right) + 1;}int* postorder(struct TreeNode* root, int* s, int* pi){if(root == NULL)return NULL;postorder(root->left,s,pi);postorder(root->right,s,pi);s[(*pi)++] = root->val;return s;}
int* postorderTraversal(struct TreeNode* root, int* returnSize) {int n = TreeSize(root);int* s = (int*)malloc(sizeof(int)*n);*returnSize = n;int i = 0;return postorder(root,s,&i);
}
  1. 另一颗树的子树。OJ链接
bool isSameTree(struct TreeNode* p, struct TreeNode* q)
{if(p == NULL && q == NULL)return true;if(p == NULL || q == NULL)return false;if(p->val != q->val)return false;return isSameTree(p->left,q->left)&& isSameTree(p->right,q->right);
}
bool isSubtree(struct TreeNode* root, struct TreeNode* subRoot){if(root == NULL)return false;if(root->val == subRoot->val){if(isSameTree(root,subRoot)){return true;}}return isSubtree(root->left,subRoot)|| isSubtree(root->right,subRoot);
}

6.二叉树的创建和销毁

二叉树的构建及遍历。OJ链接

#include <stdio.h>
#include<stdlib.h>struct TreeNode
{char val;struct TreeNode* left;struct TreeNode* right;
};
struct TreeNode* CreatBinaryTree(char* s, int* pi)
{if(s[*pi] == '#'){(*pi)++;return NULL;}struct TreeNode* root = (struct TreeNode*)malloc(sizeof(struct TreeNode));root->val = s[(*pi)++];root->left = CreatBinaryTree(s, pi);root->right = CreatBinaryTree(s, pi);return root;
}void InOrder(struct TreeNode* root)
{if(root == NULL)return;InOrder(root->left);printf("%c ",root->val);InOrder(root->right);
}int main() {char s[101];int i = 0;while(scanf("%s",s)!=EOF){struct TreeNode* root = CreatBinaryTree(s,&i);InOrder(root);}return 0;
}

6.1通过前序遍历的数组"ABD##E#H##CF##G##"构建二叉树

分析:
根据前序遍历,特别要注意的是pi,控制的是数组的小标,传进来的是指针。

BTNode* BinaryTreeCreate(BTDataType* a, int* pi)
{if (a[*pi] == '#'){(*pi)++;return NULL;}BTNode* root = (BTNode*)malloc(sizeof(BTNode));if (root == NULL){perror("malloc fail");exit(-1);}root->data = a[(*pi)++];root->left = BinaryTreeCreate(a, pi);root->right = BinaryTreeCreate(a, pi);return root;
}

6.2二叉树销毁

分析:
使用后序遍历的方式来销毁。

void BinaryTreeDestory(BTNode* root)
{if (root == NULL){return;}BinaryTreeDestory(root->left);BinaryTreeDestory(root->right);free(root);
}

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

分析:
我们知道,如果一个二叉树是完全二叉树,那么在层序遍历时,当遍历到第一个空时,如果后面的内容仍然是空的,那么这个二叉树是完全二叉树,反之则不是。

bool BinaryTreeComplete(TreeNode* root)
{Queue q;QueueInit(&q);if (root)QueuePush(&q, root);while (!QueueEmpty(&q)){TreeNode* front = QueueFront(&q);QueuePop(&q);if (front == NULL)break;QueuePush(&q, front->left);QueuePush(&q, front->right);}while (!QueueEmpty(&q)){TreeNode* front = QueueFront(&q);QueuePop(&q);if (front)return false;QueuePush(&q, front->left);QueuePush(&q, front->right);}QueueDestroy(&q);return true;
}

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

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

相关文章

贪心算法的“左最优“与“右最优“

1 答疑 1.1 什么是贪心算法的"左最优"与"右最优" "左最优"和"右最优"是贪心算法中的两种策略&#xff1a; 左最优 (Leftmost Greedy): 在每一步选择中&#xff0c;总是选择最左边&#xff08;最早出现的&#xff09;可行的选项。 右…

JVM,JRE,JDK的区别和联系简洁版

先看图 利用JDK&#xff08;调用JAVA API&#xff09;开发JAVA程序后&#xff0c;通过JDK中的编译程序&#xff08;javac&#xff09;将我们的文本java文件编译成JAVA字节码&#xff0c;在JRE上运行这些JAVA字节码&#xff0c;JVM解析这些字节码&#xff0c;映射到CPU指令集或…

洛谷——P1069 [NOIP2009 普及组] 细胞分裂(分解质因数,唯一分解定理)

文章目录 一、题目[NOIP2009 普及组] 细胞分裂题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 样例 #2样例输入 #2样例输出 #2 提示 二、题解1.基本思路&#xff1a;2.代码&#xff1a; 一、题目 [NOIP2009 普及组] 细胞分裂 题目描述 Hanks 博士是 BT&#xff08;…

粒子群算法优化支持向量SVM的供热量预测,粒子群优化支持向量机SVM回归分析

目录 背影 支持向量机SVM的详细原理 SVM的定义 SVM理论 粒子群算法原理 SVM应用实例,粒子群算法优化支持向量SVM的供热量预测,粒子群优化支持向量机SVM回归分析 代码 结果分析 展望 完整代码:粒子群算法优化支持向量SVM的供热量预测,粒子群优化支持向量机SVM回归分析_lssv…

Spring学习 基于注解的AOP配置

5.1.创建工程 5.1.1.pom.xml <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://maven.ap…

springCould中的Config-从小白开始【10 】

目录 &#x1f32d;1.spring cloud Config是什么&#x1f636;‍&#x1f32b;️&#x1f636;‍&#x1f32b;️&#x1f636;‍&#x1f32b;️ &#x1f953;2.能干什么&#x1f636;‍&#x1f32b;️&#x1f636;‍&#x1f32b;️&#x1f636;‍&#x1f32b;️ &am…

SSH远程访问出现Permission denied(password)解决方法

首先&#xff0c;这个不是密码输错了的问题&#xff1b; 1、在主机先ping一下服务器 ping XXX.XXX.XX.XXX (服务器ip地址) 如果pin成功了&#xff0c;说明可以进行连接 查看服务器的ip ifconfig2、主机连接服务器 &#xff08;服务器的ip&#xff09; ssh testXXX.XXX.XX.…

mysql 删除查询语句所选出的数据 SQL查询,用于找出outbox表中memo1字段重复且dt字段不是最新日期的记录

编写一个SQL查询&#xff0c;用于找出outbox表中memo1字段重复且dt字段不是最新日期的记录 批量打印包装箱时&#xff0c;提示有重复N条的处理方法&#xff1a; –先备份数据库&#xff0c;删除不可逆。查出来&#xff0c;如果就是重复的N条&#xff0c;全选右键 删除 SELECT …

LeetCode-数组-双指针-中等难度

文章目录 双指针1. 删除有序数组中的重复项&#xff08;入门&#xff09;1.1 题目描述1.2 解题思路1.3 代码实现 2. 删除有序数组中的重复项 II&#xff08;简单&#xff09;2.1 题目描述2.2 解题思路2.3 代码实现 3. 移动零&#xff08;简单&#xff09;3.1 题目描述3.2 代码实…

Hive的时间处理函数from_unixtime和unix_timestamp

一、概述 hive时间处理函数from_unixtime和unix_timestamp的实现以及实例&#xff0c;从而方便后续的时间处理。 二、具体功能实现 1.unix_timestamp(date[, pattern])&#xff1a; 默认的时间格式是yyyy-MM-dd HH:mm:ss&#xff0c;如果日期不是这种格式无法识别&#xff…

【技术选型】clickhouse vs starRocks

比对结论 如果只能单机部署的话&#xff0c;clickhouse基本无敌。 如果集群化&#xff0c;starRocks可以替换clickhouse&#xff0c;但支持的函数会相对少一些&#xff08;clickhouse有不少自定义函数&#xff09; 信息比对 功能clickhousestarRocksjoin大表关联容易OOM对jo…

AQS应用之BlockingQueue详解

概要 AQS全称是 AbstractQueuedSynchronizer&#xff0c;中文译为抽象队列式同步器。BlockingQueue&#xff0c;是java.util.concurrent 包提供的用于解决并发生产者 - 消费者问题的最有用的类&#xff0c;它的特性是在任意时刻只有一个线程可以进行take或者put操作&#xff0…

MySQL之导入导出远程备份

目录 一. navicat导入导出 二. mysqldump命令导入导出 导入 导出 三. load data infile命令导入导出 导入 导出 四. 远程备份 导入 导出 思维导图 一. navicat导入导出 导入&#xff1a;右键➡运行SQL文件 导出&#xff1a;选中要导出的表➡右键➡转储SQL文件➡数据和结…

C# Emgu.CV4.8.0读取rtsp流录制mp4可分段保存

【官方框架地址】 https://github.com/emgucv/emgucv 【算法介绍】 EMGU CV&#xff08;Emgu Computer Vision&#xff09;是一个开源的、基于.NET框架的计算机视觉库&#xff0c;它提供了对OpenCV&#xff08;开源计算机视觉库&#xff09;的封装。EMGU CV使得在.NET应用程序…

x-cmd pkg | vhs - 将终端的操作过程录制成视频文件的终端录制工具

目录 简介首次用户声明式录制脚本其他功能竞品和相关作品进一步阅读 简介 vhs 是一个命令行录制工具&#xff0c;用于将终端的操作过程录制成视频文件。是由 Charmbracelet 团队使用 Go 开发的&#xff0c;首个版本发布于 2022 年 10 月。开源不到一个月有接近 8k 的 star。 …

Kubernetes (八) 金丝雀发布

一. 金丝雀发布作用&#xff1a; 金丝雀发布是指在生产环境中逐步推出新版本应用程序&#xff0c;只在一小部分用户或流量中使用该版本&#xff0c;并根据反馈逐步扩…

收到的字符串写入xml并且将这个xml写入.zip文件中

文章目录 1、将数据写入xml文件WriteToXmlFile2、将xml文件写入zip压缩文件AddToZip3、组合起来4、使用到的头文件和动态库 1、将数据写入xml文件WriteToXmlFile void CSMSLoginDlg::WriteToXmlFile(const std::string& responseData, const std::string& xmlFileName…

计算机为什么有趣?哪些地方有趣?为什么学习四则运算简单,学习微积分却很难?导数是微分吗?

计算机为什么有趣&#xff1f;哪些地方有趣&#xff1f; 计算机之所以有趣&#xff0c;主要是因为它们具有无限的可能性和创造力。这里有几个方面可以帮助你理解为什么计算机这么有趣&#xff1a; 解决问题的工具&#xff1a;想象一下&#xff0c;你有一个拼图&#xff0c;计算…

PostgreSQL 支持的字段类型

PostgreSQL 支持多种字段类型&#xff0c;以下是 PostgreSQL 13 版本中支持的所有字段类型&#xff1a; 数值类型&#xff1a; smallint&#xff1a;小整数类型。integer&#xff1a;整数类型。bigint&#xff1a;大整数类型。decimal&#xff1a;精确小数类型。numeric&#x…

Redis(三)持久化

文章目录 RDB&#xff08;Redis Database&#xff09;自动触发保存频率修改dump文件保存路径修改文件保存名称dump恢复 手动触发save![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/a56fdff44aee4efa96c2ce3615b69dc1.png)bgsave 优劣优点缺点 检查修复dump文件会触…