【C++课程学习】:二叉树的基本函数实现

news/2024/10/11 7:35:29/

🎁个人主页:我们的五年

🔍系列专栏:C++课程学习

🎉欢迎大家点赞👍评论📝收藏⭐文章

 

目录

🍉二叉树的结构类型:

🍉1.创建二叉树函数(根据数组,前序遍历创建二叉树):

🍉2.销毁二叉树函数:

🍉3.前序遍历函数:

🍉4.二叉树的节点个数函数:

🍉5.计算二叉树叶子节点的个数函数:

 🍉6.计算第k层的节点个数:

🍉7.查找节点为k的元素,返回这个元素的指针

🍉8.层序遍历,借助队列:

🍉判断是否为完全二叉树:


前言:

学了那么久的二叉树,现在基本的二叉树学的差不多了,现在就给大家带来二叉树的几个基本函数。函数有几个,但是基本都不难,基本就是用了分治,递归的思想,画递归展开图也是一种很好理解递归过程好方法,等熟悉以后,就对递归有了更深的理解,面对有一些问题就直接可以写出来。

🍉二叉树的结构类型:

//二叉树的节点结构类型
typedef struct BinaryTreeNode {
    BTDataType data;
    struct BinaryTreeNode* left;        //左孩子节点的地址
    struct BinaryTreeNode* right;        //右孩子的地址
}BTNode;

每个父节点都包:

1.含存储的数据。

2.左孩子地址。

3.右孩子节点的地址。

🍉1.创建二叉树函数(根据数组,前序遍历创建二叉树):

画递归展开图也是很好理解的,先创建父亲节点,然后往左走,遇到‘#’,就返回NULL,返回上一层。

//根据数组创建二叉树,下面举例的是字符数组,创建的顺序是前序遍历
BTNode* BinaryTreeCreate(BTDataType* a, int* pi)
{if (a[*pi] == '#'){(*pi)++;return NULL;}BTNode* root = (BTNode*)malloc(sizeof(BTNode));root->data = a[*pi];	//前序遍历,先创建中间根节点(*pi)++;root->left = BinaryTreeCreate(a,pi);	//左子树root->right = BinaryTreeCreate(a,pi);	//右子树return root;	//返回root节点
}

🍉2.销毁二叉树函数:

销毁二叉树也可以前序遍历删除,也可以中序删除。不过如果是前序删除,就要在先保存左右孩子的节点。如果是中序删除,就是要保存右节点。

只有后序遍历删除不要保存节点。

// 二叉树销毁
void BTDestory(BTNode** root)
{if (*root == NULL)return;//后序遍历销毁二叉树BTDestory(&(*root)->left);BTDestory(&(*root)->right);free(*root);*root = NULL;
}

🍉3.前序遍历函数:

前序遍历和中序遍历和后序遍历基本差不多,只有后面的两个函数和打印的顺序不一样。

//前序遍历
void BTPreOreder(BTNode* root)
{if (root == NULL){printf("# ");return;}printf("%c ",root->data);BTPreOreder(root->left);BTPreOreder(root->right);
}

🍉4.二叉树的节点个数函数:

分治思想也是yyds

// 二叉树节点个数
int BTSize(BTNode* root)
{if (root == NULL)return 0;//个数等于左树节点个数+右树节点个数+1return BTSize(root->left)+BTSize(root->right)+1;
}

🍉5.计算二叉树叶子节点的个数函数:

分治

// 二叉树叶子节点个数
int BinaryTreeLeafSize(BTNode* root)
{	if (root == NULL)return 0;if (root->left == NULL && root->right == NULL)return 1;return BinaryTreeLeafSize(root->left) + BinaryTreeLeafSize(root->right);
}

 🍉6.计算第k层的节点个数:

要注意的是,遇到NULL,要返回,还有就是,k==1,return 1,要在后面,因为如果在前面,k确实等于1。但是这时候是空节点,所以不能return 1,所以return 1要在后面。

// 二叉树第k层节点个数
int BinaryTreeLevelKSize(BTNode* root, int k)
{if (root == NULL)return 0;if (k == 1){return 1;}return BinaryTreeLevelKSize(root->left, k - 1)+ BinaryTreeLevelKSize(root->right,k-1);
}

🍉7.查找节点为k的元素,返回这个元素的指针

找父节点,父节点不是,就去左树,左树没有,就去右树。

只要找到了就返回,所以是或的关系;

// 二叉树查找值为x的节点
BTNode* BinaryTreeFind(BTNode* root, BTDataType x)
{if (root == NULL)return NULL;if (root->data == x)return root;BTNode* p1 = BinaryTreeFind(root->left, x);if (p1!=NULL)return p1;BTNode* p2 = BinaryTreeFind(root->right, x);if (p2!=NULL)return p2;return NULL;
}

🍉8.层序遍历,借助队列:

先在队列中插入root,在队列头出一个数据,就入这个节点的左右孩子节点。因为队列有先进先出的特点,所以能达到层序的目的。

// 层序遍历
void BinaryTreeLevelOrder(BTNode* root)
{Queue ps;QueueInit(&ps);QueuePush(&ps, root);while (!QueueEmpty(&ps)){BTNode* node = QueueTop(&ps);if (node == NULL){break;}else{printf("%c ", node->data);		QueuePush(&ps, node->left);QueuePush(&ps, node->right);}QueuePop(&ps);}QueueDestroy(&ps);
}

🍉9.判断是否为完全二叉树:

先入数据,然后出,和层序遍历一样,当需要NULL就结束。然后看后面的队列是否都是NULL,如果只要有一个不是NULL,那肯定就不是完全二叉树了,前面都有NULL,后面又出现节点。

// 判断二叉树是否是完全二叉树
int BinaryTreeComplete(BTNode* root)
{Queue ps;QueueInit(&ps);QueuePush(&ps, root);while (!QueueEmpty(&ps)){BTNode* node = QueueTop(&ps);if (node == NULL){break;}else{QueuePush(&ps, node->left);QueuePush(&ps, node->right);}QueuePop(&ps);}while (!QueueEmpty(&ps)){BTNode* node = QueueTop(&ps);if (node != NULL)return 0;QueuePop(&ps);}return 1;
}


http://www.ppmy.cn/news/1464043.html

相关文章

谈谈关于mysql索引的理解

索引 我们在学习java中用来表示数组的下标例如定义一个变量int i 这就表示一个索引,因为索引的英文单词是index,索引也可以称为是书的目录,它可以方便我们查询自己所需要的内容,通过索引我们可以快速找到自己的需求.此时引出了索引的概念,在数据库中. 关于索引的相关操作 有…

【Nginx <三>⭐️⭐️⭐️】Nginx 负载均衡使用

目录 👋前言 👀一、 负载均衡概述 🌱二、项目模拟 2.1 环境准备 2.2 启动多个服务器 2.3 配置 Nginx 2.4 测试配置 💞️三、章末 👋前言 小伙伴们大家好,前不久开始学习了 Nginx 的使用,在…

docker image prune -f 命令什么用途

docker image prune -f 命令用于清理系统中未被使用的 Docker 镜像。具体来说,它会删除那些未被任何容器使用的悬空镜像(dangling images),从而释放磁盘空间。 以下是 docker image prune -f 命令的具体用途和作用: …

mac安装的VMware虚拟机进行桥接模式配置

1、先进行网络适配器选择,选择桥接模式 2、点击网络适配器 设置... 3、选择WiFi(我使用的是WiFi,所以选择这个),注意看右边的信息:IP和子网掩码,后续配置虚拟机的ifcfg-ens文件会用到 4、编辑if…

贪心算法[1]

首先用最最最经典的部分背包问题来引入贪心的思想。 由题意可知我们需要挑选出价值最大的物品放入背包&#xff0c;价值即单位价值。 我们需要计算出每一堆金币中单位价值。金币的属性涉及两个特征&#xff0c;重量和价值。 所以我们使用结构体。 上代码。 #include <i…

1.每日设计模式-理论

目录 一、什么是设计模式 二、设计原则 三、设计模式的种类 代码地址&#xff1a;patterns: 每日设计模式 一、什么是设计模式 软件设计模式(Design Pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结&#xff0c;使用设计模式是为了可重用代码…

VSCode开发Python-Django入门

一、安装配置Python环境及配置Python环境变量 1、python安装包安装后&#xff0c;需要注意pip.exe和pip3.exe的安装&#xff1b; 2、环境变量需要配置两个目录&#xff1b; 3、验证python是否安装成功 通过cmd命令执行&#xff1a;python --version 查看python版本&#xff…

【全开源】赛事报名系统源码(Fastadmin+ThinkPHP和Uniapp)

基于FastadminThinkPHP和Uniapp开发的赛事报名系统&#xff0c;包含个人报名和团队报名、成绩查询、成绩证书等。 构建高效便捷的赛事参与平台 一、引言&#xff1a;赛事报名系统的重要性 在举办各类赛事时&#xff0c;一个高效便捷的报名系统对于组织者和参与者来说都至关重…