小白的入门二叉树(C语言实现)

news2025/1/11 7:54:42

前言:

二叉树属于数据结构的一个重要组成部分,很多小白可能被其复杂的外表所吓退,但我要告诉你的是“世上无难事,只怕有心人”,我将认真的对待这篇博客,我相信只要大家敢于思考,肯定会有所收获的,当我们攀过一座山,回头看去,可能当初畏惧的大山也不过如此。

目录

前言:

一,树的基本知识

1树的概念

2,树相关概念

二,二叉树的基本知识

1,二叉树的概念

2,特殊的二叉树:

三,二叉树的思路及代码实现(边讲解边写代码)

1,二叉树实现方式思考

2,前提准备(头文件和二叉树基本结构)

3,如何创建二叉树

4,二叉树的前,中,后序遍历

5,层序遍历

6,二叉树的拓展——堆(堆排序,TOPk问题)

1,堆的概念

2,堆的代码实现思考

3,向下排序

4,向上排序

5,Topk问题

1,建堆

2,堆如何插入呢?

3,堆顶数据如何删除呢?


一,树的基本知识

1树的概念


树是一种非线性的数据结构,它是由n(n>=0)个有限结点组成一个具有层次关系的集合。把它叫做树是因为它看起来像一棵倒挂的树,也就是说它是根朝上,而叶朝下的。
有一个特殊的结点,称为根结点,根节点没有前驱结点
除根节点外,其余结点被分成M(M>0)个互不相交的集合T1、T2、……、Tm,其中每一个集合Ti(1<= i 
<= m)又是一棵结构与树类似的子树。每棵子树的根结点有且只有一个前驱,可以有0个或多个后继因此,树是递归定义的。

注意:树形结构中,子树之间不能有交集,否则就不是树形结构

2,树相关概念

节点的度:一个节点含有的子树的个数称为该节点的度; 如上图:A的为6
叶节点或终端节点:度为0的节点称为叶节点; 如上图:B、C、H、I...等节点为叶节点
非终端节点或分支节点:度不为0的节点; 如上图:D、E、F、G...等节点为分支节点
双亲节点或父节点:若一个节点含有子节点,则这个节点称为其子节点的父节点; 如上图:A是B的父节点孩子节点或子节点:一个节点含有的子树的根节点称为该节点的子节点; 如上图:B是A的孩子节点
兄弟节点:具有相同父节点的节点互称为兄弟节点; 如上图:B、C是兄弟节点
树的度:一棵树中,最大的节点的度称为树的度; 如上图:树的度为6
节点的层次:从根开始定义起,根为第1层,根的子节点为第2层,以此类推;
树的高度或深度:树中节点的最大层次; 如上图:树的高度为4
堂兄弟节点:双亲在同一层的节点互为堂兄弟;如上图:H、I互为兄弟节点
节点的祖先:从根到该节点所经分支上的所有节点;如上图:A是所有节点的祖先
子孙:以某节点为根的子树中任一节点都称为该节点的子孙。如上图:所有节点都是A的子孙
森林:由m(m>0)棵互不相交的树的集合称为森林;

二,二叉树的基本知识

1,二叉树的概念

一棵二叉树是结点的一个有限集合,该集合:
1. 或者为空
2. 由一个根节点加上两棵别称为左子树和右子树的二叉树组成

从上图可以看出:
1. 二叉树不存在度大于2的结点
2. 二叉树的子树有左右之分,次序不能颠倒,因此二叉树是有序树
注意:对于任意的二叉树都是由以下几种情况复合而成的:

2,特殊的二叉树:


1. 满二叉树:一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是满二叉树。也就是
说,如果一个二叉树的层数为K,且结点总数是 ,则它就是满二叉树。
2. 完全二叉树:完全二叉树是效率很高的数据结构,完全二叉树是由满二叉树而引出来的。对于深度为K
的,有n个结点的二叉树,当且仅当其每一个结点都与深度为K的满二叉树中编号从1至n的结点一一对应时称之为完全二叉树。 要注意的是满二叉树是一种特殊的完全二叉树。

三,二叉树的思路及代码实现(边讲解边写代码)

1,二叉树实现方式思考

想要用代码实现二叉树,我们首先要思考二叉树的结构特性,二叉树最多只有两个孩子,我们抓住这个关键特性,我们有两个实现方式:链表或者数组,我们只讲一种实现方式,链表实现二叉树,当然我讲完链表实现后,数组实现也不过是小菜一碟,现在正菜开始。

2,前提准备(头文件和二叉树基本结构)
#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<stdbool.h>
typedef char BTDataType;//BTDataType是char的别名,方便以后修改二叉树的数据类型

typedef struct BinaryTreeNode
{
	BTDataType _data;              //二叉树的存储数据
	struct BinaryTreeNode* _left;  //左孩子
	struct BinaryTreeNode* _right; //右孩子
}BTNode;
3,如何创建二叉树

要求:给你一串字符,要求你根据字符前序遍历创建二叉树

字符:"ABD##E#H##CF##G##"

利用字符创建二叉树,一开始我们肯定是一头雾水,我们首先自己在纸上试试创建二叉树

不知道大家有没有发现规律,因为是前序遍历,因此是对根A进行赋值,然后是左孩子B,像C如果左孩子是NULL,就返回上一个根C对右孩子进行赋值,右孩子如果非空,就继续对右孩子的左孩子进行赋值,遇到空就返回对右孩子的右孩子赋值,直到全为空就返回到B。

那么这个思路我们该怎么实现呢,不知道大家有没有发现这个可以拆分为一个个类似的子问题,那么结果就呼之欲出了-----递归。接下来跟上代码继续讲解一遍思路


BTNode* TreeCreat(BTDataType* a, int* pi) {
	if (a[*pi] == '#') {	//如果为空就返回
		(*pi)++;			//即使为空也要继续下一个字母
		return NULL;		//提前结束这一层递归
	}
	BTNode* binary = (BTNode*)malloc(sizeof(BTNode));//分配空间
	binary->_data = a[*pi];			//将值放进data中
	(*pi)++;                        //读完,指向下一个字母
	binary->_left = TreeCreat(a, pi);	//左孩子进行赋值,直到遇空返回
	binary->_right  = TreeCreat(a, pi);//右孩子进行赋值,直到遇空返回
	return binary;    //返回创建的结点给上一层递归
}
BTNode* BinaryTreeCreate(BTDataType* a,int n, int* pi) {
	*pi = 0;					//用这个来记录读取的数据到哪里了
	return TreeCreat(a,pi);		//开始递归读值
}

4,二叉树的前,中,后序遍历

二叉树前,中,后遍历呢?大家先想想前序创建二叉树,有没有发现遍历只要按照创建时的思路就行了,那么话不多说·直接上前序代码

void BinaryTreePrevOrder(BTNode* root) {
	if (root == NULL) {   //老规矩,遇空返回
		printf("#");		
		return;
	}
	printf("%c", root->_data);		//按照前序顺序先打印根
	BinaryTreePrevOrder(root->_left); //开始进入左孩子,进入之后会打印左孩子,一直走到NULL
	BinaryTreePrevOrder(root->_right);//开始进入右孩子,进入之后会打印左孩子,一直走到NULL
}

中序代码

void BinaryTreeInOrder(BTNode* root) {
	if (root == NULL) {  //防止非法访问,直接判断是否为空
		printf("#");
		return;
	}
	BinaryTreeInOrder(root->_left);	//进入递归,先直到最左的叶结点
	printf("%c", root->_data);     //开始打印
	BinaryTreeInOrder(root->_right);//根和左孩子都遍历了开始进入右孩子并打印
}

相信聪明的你可以独自写出后序遍历了吧!!!

5,层序遍历

前中后遍历还有迹可寻,那层序遍历应该怎么入手呢?层序遍历的如果光按照树的特性是不太好入手的,但是我们想一下,假设打印根节点,再打印它的两个孩子,再打印根节点孩子的所以孩子,这个中间是不是有一个顺序关系,你有没有想到队列呢,这样的话是不是可以依次拿到数据呢?在里面还有一个重要的关系,就是一个父亲结点会有两个孩子,也就意味着入队列的速度是远大于出队列的速度的,想完这些我们就可以写代码了。想看队列的源代码你们可以看我的往期文章。

void BinaryTreeLevelOrder(BTNode* root) {
	Queue* q = (Queue*)malloc(sizeof(Queue));//创建队列
	QueueInit(q);							//初始化队列
	QueuePush(q, root);						//先将根入队列
	while (q->_front != q->_rear ) {        //这个时候队列头等于尾,意味着队列为空了
		if (q->_front->_data   != NULL) {	//入队列第一个非空元素的两个孩子
			QueuePush(q, q->_front->_data->_left);
			QueuePush(q, q->_front->_data->_right);
			printf("%c", q->_front->_data->_data);//打印队列第一个元素的休息
		}
		else
		printf("#");                          //为空也要打印
		QueuePop(q);							//出掉原先的第一个元素,第二个元素顶上
	}
}

6,二叉树的拓展——堆(堆排序,TOPk问题)

1,堆的概念

堆就是一个完全二叉树

大堆就是父亲节点一定大于孩子节点

小堆就是父亲节点小于孩子节点

2,堆的代码实现思考

堆是二叉树,我们同样可以用链表和数组实现,我们这里为了突出堆的特性和作用,我们这里用数组实现,在实现前我们有两个规则会帮助我们进行排序,大家先记住下标规律,规则的条件是,根节点在数组中的位置为1,孩子节点是按顺序存储的,先左孩子,后右孩子

左孩子下标=父节点下标/2+1

右孩子下标=父节点下标/2+2、

父节点下标=(孩子节点下标-1)/2      //因为int是没有小数的,1/2=0.所以左右节点没必要区分

3,向下排序

首先声明,堆排序不是绝对有序,而是父节点相比孩子节点是有序的,但兄弟节点和同一层节点不一定有序。

向下排序是如何排的呢?顾名思义,就是从根节点开始和孩子节点比较,如果大小不符合堆的特性,就交换位置,直到堆的叶结点,如果大小符号就直接结束,开始下一层排序,我们要利用父节点和孩子节点的位置才能进行比较和交换

接下来看一下小堆排序的图

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
typedef int HPDataType;
typedef struct Heap
{
	HPDataType* _a;
	int _size;  //堆的大小
	int _capacity;//堆的总容量
}Heap;
void Adjustdown(HPDataType* a, int n,int capcity) {
	int parent = n;			//参数传来排序的位置下标
		int child = parent * 2 + 1;			//找到孩子节点下标
		while (child < capcity) {          //防止数组越界
			if (child + 1 < capcity && a[parent] > a[child + 1] && a[child + 1] < a[child]) {
				swap(a + parent, a + child + 1);  //交换
				parent = child + 1;              //父子换为左孩子字节的下标,开始下一轮交换
			}
			else if (a[parent] > a[child]) {
				swap(a + parent, a + child);
				parent = child;					//父子换为右孩子字节的下标,开始下一轮交换
			} 
			else
				break;							//无需交换,直接结束
			child = parent * 2 + 1;             //更新孩子节点的位置下标
		}
}
4,向上排序

向上排序和向下排序相反,它是从子节点到父节点,到根节点就会强制终止,

void Adjustup(HPDataType* a, int n) {
	int parent = (n - 1) / 2;//通过参数找到父节点
	int child = n;			//找到孩子节点
	while (child > 0) {
		if (a[child] < a[parent])		
			swap(a + child, a + parent);
		else
			break;			//如果不需要交换就停止,前提是堆是有序的
		child = parent;   //将子节点换位父节点,向上排序
		parent = (child - 1) / 2;		//父节点继续向上
	}
}

现在你有没有想为什么写向上排序和向下排序,其实向上排序和向下排序就像插入排序和冒泡排序的作用,只不过是用于堆的排序,但想要将一个毫无规律的堆排序是需要很多个向上排序和向下排序的,我们可以利用循坏来进行重复的排序。

5,Topk问题

堆排序的一个很重要的作用就是Topk问题,简而言之就是排行榜,相比与冒泡的o(n^2),堆排序的时间复杂度是nlogn。

1,建堆

首先要求Topk,我们需要一个容量为k的堆,这样我们才能进行排序,因此我们首先来建堆,并且将堆进行排序

void HeapCreate(Heap* hp, HPDataType* a, int n) {
	assert(hp);													//防止非法访问导致程序崩溃
	hp->_a = (HPDataType*)malloc(sizeof(HPDataType) * n);		//开辟堆空间
	assert(hp->_a);												//判断是否开辟成功
	for (int i = 0; i < n; i++)
		hp->_a[i] = a[i];										//将数据全部输入堆,注意此时无序
	hp->_capacity = n;											//记录容量
	hp->_size = n;												//记录大小
	for(int i=(n-2)/2;i>=0;i--)									//找到第一个倒数的父亲节点,并且逐个向下排序直到根
	Adjustdown(hp->_a ,i,n);
	for (int i = 0; i < n; i++)
		printf("%d ", hp->_a[i]);								//打印方便观察
}
2,堆如何插入呢?

从数组的角度来看,我们可以先不管大小关系,直接将新数据放到末尾再来考虑顺序问题,将数据放到末尾之后排序就简单了,我们直接用一个向上排序就可以是堆再次有序了

void HeapPush(Heap* hp, HPDataType x) {
	if (hp->_capacity == hp->_size) {     //检查堆的剩余空间
		HPDataType* b;
		b= (HPDataType*)realloc(hp->_a ,sizeof(HPDataType) * (hp->_capacity + 3));  //一次开辟三个内存
		assert(b);						//判断是否开辟成功
		hp->_a = b;						//将开辟好的堆放入原位置
		hp->_capacity += 3;      //调整堆的容量记录
	}
	hp->_a[hp->_size] = x;					//放到数组末尾
	Adjustup(hp->_a, hp->_size);				//向上调整一次就有序了
	hp->_size++;
}
3,堆顶数据如何删除呢?

堆顶数据的删除看似是一个复杂的问题,我们要想清楚,如果直接把堆顶删除,然后之后的数据就依次往前推,但是会有一个问题,往前推一定是有序的吗?我们之前讲了,堆同层是不一定有序的,如果随意的往前覆盖,就会导致原先同层小的数据成为了父节点,此时会产生蝴蝶效应,我们无法判断有多少个无序了,因此我们要另辟蹊径,想一想如果我们把第一个时间和最后一个数据交换,然后把最后一个数据无视,那么是不是只要根节点的数据是无序的,那么此时我们只需要堆根进行一次向下排序不就有序了吗?

void HeapPop(Heap* hp) {
	assert(hp);   //防止非法访问
	assert(hp->_a);
	swap(hp->_a+0, hp->_a+hp->_size -1);//交换第一个和最后一个数据
	Adjustdown(hp->_a, 0, hp->_size-1);//直接进行一次向下排序完成
	hp->_size--;
}

博客耗时颇多,希望大家点赞加收藏,可以评论区交流

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

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

相关文章

Stm32_标准库_1

代码&#xff1a; #include "stm32f10x.h" // Device headerGPIO_InitTypeDef GPIO_InitStructure;//定义变量结构体int main(void){/*使用RCC开启GPIO的时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//开启PA端口时钟/*使用GPIO_…

折线图geom_line()参数选项

往期折线图教程 图形复现| 使用R语言绘制折线图折线图指定位置标记折线图形状更改 | 绘制动态折线图跟着NC学作图 | 使用python绘制折线图 前言 我们折线的专栏推出一段时间&#xff0c;但是由于个人的原因&#xff0c;一直未进行更新。那么今天&#xff0c;我们也参考《R语…

(循环)mysql定时器删除某表中数据例子

CREATE EVENT clear_interactive_logs ON SCHEDULE EVERY 1 DAY STARTS 2023-09-21 23:36:36 DO DELETE from t_interactive_log WHERE id not IN (SELECT * from (SELECT id from t_interactive_log ORDER BY occer_time DESC limit 20000) x ); END ———————————…

Spring Boot魔法:简化Java应用的开发与部署

文章目录 什么是Spring Boot&#xff1f;1. 自动配置&#xff08;Auto-Configuration&#xff09;2. 独立运行&#xff08;Standalone&#xff09;3. 生产就绪&#xff08;Production Ready&#xff09;4. 大量的起步依赖&#xff08;Starter Dependencies&#xff09; Spring …

QT实现qq登录

1、登录界面 头文件 #ifndef MAINWINDOW_H #define MAINWINDOW_H#include <QMainWindow> #include <QMessageBox> #include <QDebug> #include "second.h" //第二个界面头文件 #include "third.h" //注册界面头文件#include <QSq…

如何进行性能测试

文章目录 前言什么是性能测试为什么要做性能测试怎么做我们的性能测试SoloPiSoloPi的介绍和安装SoloPi的性能数据 前言 随着科学技术的迅速发展&#xff0c;信息时代离不开软件&#xff0c;软件的成功上线离不开软件测试的功劳&#xff0c;因此软件测试对于软件的重要性不言而…

最新Java JDK 21:全面解析与新特性探讨

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

STP生成树协议基本配置示例---STP逻辑树产生和修改

STP是用来避免数据链路层出现逻辑环路的协议&#xff0c;运行STP协议的设备通过交互信息发现环路&#xff0c;并通过阻塞特定端口&#xff0c;最终将网络结构修剪成无环路的树形结构。在网络出现故障的时候&#xff0c;STP能快速发现链路故障&#xff0c;并尽快找出另外一条路径…

放大电路的理解

如有错误&#xff0c;敬请指正 【电子】模拟电子技术基础 上交大 郑益慧主讲&#xff08;模拟电路/模电 讲课水平堪比华成英&#xff0c;视频质量完爆清华版&#xff09;_哔哩哔哩_bilibili

Redis面试问题三什么是缓存雪崩怎么解决

定义 缓存雪崩是因为大量的key设置了同一过期时间的导致在同一时间类缓存同时过期&#xff0c;而这时因为请求过来已经没有缓存了&#xff0c;DB压力大数据库崩溃了。 解决方法 我可以在设置过期时间的时候加一个随机时间&#xff0c;在1-5分钟那样可以分散过期时间&#xf…

ClickHouse面向列的数据库管理系统(原理简略理解)

目录 官网 什么是Clickhouse 什么是OLAP 面向列的数据库与面向行的数据库 特点 为什么面向列的数据库在OLAP场景中工作得更好 为什么ClickHouse这么快 真实的处理分析查询 OLAP场景的关键属性 引擎作用 ClickHouse引擎 输入/输出 CPU 官网 https://clickhouse.com…

python实现命令tree的效果

把所有的文档都传到了git上,但是内容过多找起来不方便,突发奇想如果能在readme中,递归列出所有文件同时添加上对应的地址,这样只需要搜索到对应的文件点击就能跳转过去了… 列出文件总得有个显示格式,所以就按照tree的来了… 用python实现命令tree的效果 首先,这是tree的效果…

JS 手写call、apply和bind方法

手写call、apply和bind方法 一、方法介绍1.1 call 方法1.2 apply 方法1.3 bind 二、方法的实现2.1 call 方法2.2 apply 方法2.3 bind 方法 一、方法介绍 apply、call和bind都是系统提供给我们的内置方法&#xff0c;每个函数都可以使用这三种方法&#xff0c;是因为apply、call…

Unity中关于多线程的一些事

一.线程中不允许调用unity组件api 解决方法&#xff1a;可以使用bool值变化并且在update中监测bool值变化来调用关于unity组件的API. 二.打印并且将信息输出到list列表中 多线程可能同时输出多条信息。输出字符串可以放入Queue队列中。队列可以被多线程插入。 三.启用socke…

计算机基础协议/概念:推送数据— —WebSocket与SSE;前端Blob/URL下载文件

计算机基础协议/概念&#xff1a;推送数据— —WebSocket与SSE 1 WebSocket&#xff1a;双向通信 1.1 概念&#xff1a;通信过程 ①Upgrade&#xff1a;浏览器告知服务器升级为WebSocket协议 ②Switch&#xff1a;服务器升级成功后会返回101状态码 ③Communicate&#xff1…

SQL注入脚本编写

文章目录 布尔盲注脚本延时注入脚本 安装xampp&#xff0c;在conf目录下修改它的http配置文件&#xff0c;如下&#xff0c;找到配置文件&#xff1a; 修改配置文件中的默认主页&#xff0c;让xampp能访问phpstudy的www目录&#xff0c;因为xampp的响应速度比phpstudy快得多&am…

使用EasyExcel后端导出excel

官方文档&#xff1a;关于Easyexcel | Easy Excel 这里进行简单记录&#xff0c;方便确定是不是适用此方式&#xff1a; 零&#xff1a;实体类中注解用法 一&#xff1a;读excel /*** 强制读取第三个 这里不建议 index 和 name 同时用&#xff0c;要么一个对象只用index&…

代码随想录算法训练营第二天(C) | 977.有序数组的平方 209.长度最小的子数组 59.螺旋矩阵

文章目录 前言一、977.有序数组的平方二、209.长度最小的子数组三、59.螺旋矩阵总结 前言 java版&#xff1a; 代码随想录算法训练营第二天 | 977.有序数组的平方 &#xff0c;209.长度最小的子数组 &#xff0c;59.螺旋矩阵_愚者__的博客-CSDN博客 一、977.有序数组的平方 …

Python环境配置及基础用法Pycharm库安装与背景设置及避免Venv文件夹

目录 一、Python环境部署及简单使用 1、Python下载安装 2、环境变量配置 3、检查是否安装成功 4、Python的两种模式&#xff08;编辑模式&交互模式&#xff09; 二、Pycharm库安装与背景设置 1、Python库安装 2、Pycharm自定义背景 三、如何避免Venv文件夹 一、P…

【Java 基础篇】Java TCP通信详解

TCP&#xff08;Transmission Control Protocol&#xff09;是一种面向连接的、可靠的网络传输协议&#xff0c;它提供了端到端的数据传输和可靠性保证。TCP通信适用于对数据传输的可靠性和完整性要求较高的场景&#xff0c;如文件传输、网页浏览等。本文将详细介绍Java中如何使…