数据结构之树 --- 二叉树 < 堆 >

目录

1. 树是什么?

 1.1 树的表示

2. 二叉树 

2.1 二叉树的概念

2.2 特殊的二叉树 

2.3 二叉树的性质

2.4  二叉树的存储结构

2.4.1 顺序存储

2.4.2 链式存储

3. 二叉树顺序结构的实现 <堆>

3.1 二叉树的顺序结构

​编辑 3.2 堆的概念及结构

​编辑 3.3 堆的实现(以小堆为例)

3.3.1 堆结构的定义

3.3.2 向下调整算法 <此处向下调整代码以整棵树的根节点为例 >

概念

代码展示 

3.3.3 向上调整算法

概念

代码展示  <此处向上调整代码以最后一个节点为例 >

3.3.4 堆的插入

堆的插入示例图

代码展示

3.3.5 堆的删除

堆的删除示例图

 3.4 堆的代码实现


1. 树是什么?

 数据结构中的树是一种非线性数据结构。树这个概念用来描述具有层级关系的结构。

树的数据结构的主要特征和概念包括:

- 节点(Node):树中信息的基本单位。

- 根节点(Root Node):树中位于最顶层的节点,没有父节点。

- 子节点(Child Node):相对于父节点而言的下级节点。

- 父节点(Parent Node):相对于子节点而言的上级节点。 

- 叶节点(Leaf Node):没有子节点的节点。

- 分支(Branch):连接节点的边。

- 枝(Edge):连接两个节点的关系。

- 子树(Subtree):以某个节点为根的树形结构。

- 树的高度(Height):从根节点到最远叶节点的最长路径上的边数。

- 树的度(Degree):一个节点的子节点数目。

常见的树数据结构包括二叉树、B树、平衡树、哈夫曼树等。它们通过节点和边构成了一个包含层级关系的抽象数据模型,广泛应用于文件系统、网络协议、表达式求值等领域。

树的数据结构相对线性表而言,支持有效地表达具有分层关系的结构化数据。它是理解递归和分治算法的重要基础。

如下图就是一个树结构。

切记树形结构中,子树之间不能有交集。 

 1.1 树的表示

树结构相对于线性表复杂很多,我们不但要保存树结构每一个节点的值,还要保存节点之间的关系。

例如我们使用孩子兄弟表示法:

2. 二叉树 

2.1 二叉树的概念

二叉树是一个节点构成的有限集合,该集合:

1.或者为空;

2.或者由一个根节点与两棵别称为左子树和右子树的二叉树组成。

由其概念可知,二叉树不存在度大于2的节点,且二叉树分为左右子树,不能颠倒,是有序树。因而二叉树可分为以下几种情况:

2.2 特殊的二叉树 

满二叉树:

一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是满二叉树。也就是说,如果一个二叉树的层数为K,且结点总数是 ,则它就是满二叉树。

完全二叉树:

完全二叉树是效率很高的数据结构,完全二叉树是由满二叉树而引出来的。对于深度为K
的,有n个结点的二叉树,当且仅当其每一个结点都与深度为K的满二叉树中编号从1至n的结点一一对应时称之为完全二叉树。 要注意的是满二叉树是一种特殊的完全二叉树。

话不多说,上图:

2.3 二叉树的性质

1. 若规定根节点的层数为1,则一棵非空二叉树的第i层上最多有 2^(i-1)个结点。
2. 若规定根节点的层数为1,则深度为h的二叉树的最大结点数是 2^h-1。
3. 对任何一棵二叉树, 如果度为0其叶结点个数为 , 度为2的分支结点个数为 ,则有 n0=n2+1;
4. 若规定根节点的层数为1,具有n个结点的满二叉树的深度,h= log^(n+1)。 (ps:log^(n+1) 是log以2为底,n+1为对数)
5. 对于具有n个结点的完全二叉树,如果按照从上至下从左至右的数组顺序对所有节点从0开始编号,则对于序号为i的结点有:
1. 若i>0,i位置节点的双亲序号:(i-1)/2;i=0,i为根节点编号,无双亲节点
2. 若2i+1<n,左孩子序号:2i+1,2i+1>=n否则无左孩子
3. 若2i+2<n,右孩子序号:2i+2,2i+2>=n否则无右孩子

2.4  二叉树的存储结构

二叉树的存储结构也分为顺序存储与链式存储。

2.4.1 顺序存储

顺序结构存储就是使用数组来存储,一般使用数组只适合表示完全二叉树,因为不是完全二叉树会有空间的浪费。而现实中使用中只有堆才会使用数组来存储,关于堆我们后面的章节会专门讲解。二叉树顺序存储在物理上是一个数组,在逻辑上是一颗二叉树

2.4.2 链式存储

二叉树的链式存储结构是指用链表来表示一棵二叉树,即用链来指示元素的逻辑关系。 通常的方法是链表中每个结点由三个域组成,数据域和左右指针域,左右指针分别用来给出该结点左孩子和右孩子所在的链结点的存储地址 。链式结构又分为二叉链和三叉链,目前我们只看二叉链。

左右孩子表示法:

typedef int BTDataType;
// 二叉链
struct BinaryTreeNode
{
struct BinTreeNode* pLeft; // 指向当前节点左孩子
struct BinTreeNode* pRight; // 指向当前节点右孩子
BTDataType data; // 当前节点值域
}

3. 二叉树顺序结构的实现 <堆>

3.1 二叉树的顺序结构

普通的二叉树是不适合用数组来存储的,因为可能会存在大量的空间浪费。而完全二叉树更适合使用顺序结构存储。现实中我们通常把堆(一种二叉树)使用顺序结构的数组来存储,需要注意的是这里的堆和操作系统虚拟进程地址空间中的堆是两回事,一个是数据结构,一个是操作系统中管理内存的一块区域分段。

完全二叉树使用数组存储不会由空间浪费。 

而非完全二叉树将会造成大量的空间浪费。 

 3.2 堆的概念及结构

概念:

堆是一种重要的数据结构,主要特点如下:

  • 堆是一棵完全二叉树。

  • 堆分为大堆和小堆。

  • 最大堆:每个节点的值都大于或等于其子节点的值。

  • 最小堆:每个节点的值都小于或等于其子节点的值。

  • 堆的根节点分别为最大值(最大堆)或最小值(最小堆)。

  • 堆支持两种基本操作:插入一个元素和删除根节点。

  • 插入:新元素添加到叶子节点,然后不断与父节点比较交换位置,直到符合堆的性质。时间复杂度O(logN)。

  • 删除根节点:将最后一个叶子节点移到根,然后与子节点比较交换位置重建堆。时间复杂度也是O(logN)。

  • 堆常用于优先级队列,支持快速获取最大/最小元素,以及插入和删除操作。

所以总结来说,堆是一种特殊的完全二叉树结构,能够快速支持获取最大/最小元素和插入/删除操作,广泛应用于优先级队列等数据结构中。它通过维护节点值的堆积性质来实现高效操作。

 3.3 堆的实现(以小堆为例)

3.3.1 堆结构的定义

typedef struct Heap
{
	HPDataType* a;//存放数据的数组
	int size;//数组内的元素个数
	int capacity;//数组的容量
}Heap;

要实现堆,我们就必须对向下调整算法和向上调整算法有一个明确的认知。

3.3.2 向下调整算法 <此处向下调整代码以整棵树的根节点为例 >

概念

从上向下调整,以某个节点为根节点,比较其左孩子与右孩子的大小,选择其中小的和父节点相比,如果小于父节点,则交换值,更新父节点与子节点,循环往复。

大堆则寻找最大值。

代码展示 
void ADjustdown(Heap* hp)
{
	int parent = 0;//以根节点为始
	int child = parent * 2 + 1;//求该节点的孩子,此刻计算的为左孩子
//后续比较左孩子与右孩子的大小,谁小谁做孩子
	while (child<hp->size)//左孩子存在
	{
	
		if (child + 1 < hp->size && hp->a[child + 1] < hp->a[child])//右孩子存在且小于左孩
		{
			child = child + 1;//将右节点赋值给孩子几点
		}
		if (hp->a[child] < hp->a[parent])//孩子节点的值小于父节点
		{
			swap(&hp->a[child], &hp->a[parent]);//交换父节点与孩子节点data
			parent = child;//更新父节点的下标位置
			child = parent * 2 + 1;//计算下一轮孩子节点的下标
		}
		else
			break;
	}
}

3.3.3 向上调整算法

概念

从下向上调整,以某个节点为子节点,比较该节点与父节点的大小,如果小于父节点,则交换并更新父子节点。大堆则相反。

代码展示  <此处向上调整代码以最后一个节点为例 >
void ADjustup(Heap* hp)
{
	int child = hp->size - 1;//size是元素个数,所以最后一个元素的下标为size-1
    //以最后一个元素为始
	int parent = (child - 1) / 2;//求其父节点
	while (child > 0)//孩子节点存在
	{
		if (hp->a[child] < hp->a[parent])//如果父节点大于子节点,则交换,并更新父子结点
		{
			swap(&(hp->a[child]), &(hp->a[parent]));
			child = parent;
			parent = (parent - 1) / 2;
		}
		else
			break;
	}
}

3.3.4 堆的插入

堆的插入是在堆尾插入,然后借用向上调整算法,直到满足堆。

堆的插入示例图

代码展示
void HeapPush(Heap* hp, HPDataType x)
{
	assert(hp);
	assert(hp->a);

	if (hp->size == hp->capacity)//如果堆已满,则进行扩容
	{
		int newcapacity = hp->capacity == 0 ? 4 : hp->capacity * 2;
		HPDataType* tmp = realloc(hp->a, sizeof(HPDataType) * newcapacity);
		assert(tmp);
		hp->a = tmp;
		hp->capacity = newcapacity;
	}

	hp->a[hp->size] = x;
	hp->size++;
	ADjustup(hp);//向上调整
}

3.3.5 堆的删除

堆的删除是删除堆的根节点,即交换堆顶与堆尾元素,然后删除堆尾,再进行向下调整算法。

堆的删除示例图

代码展示

void HeapPop(Heap* hp)
{
	assert(hp);
	assert(hp->size > 0);
	swap(&(hp->a[0]), &(hp->a[hp->size - 1]));
	hp->size--;
	ADjustdown(hp);
}

 3.4 堆的代码实现

Heap.h

#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 HeapCreate(Heap* hp, int n);

// 堆的销毁
void HeapDestory(Heap* hp);

// 堆的插入
void HeapPush(Heap* hp, HPDataType x);

// 堆的删除
void HeapPop(Heap* hp);

// 取堆顶的数据
HPDataType HeapTop(Heap* hp);

// 堆的数据个数
int HeapSize(Heap* hp);

// 堆的判空
bool HeapEmpty(Heap* hp);

Heap.c

#include"heap.h"
void HeapCreate(Heap* hp, int n)
{
	assert(hp);
	hp->size = 0;
    hp->capacity = n;
	hp->a = (HPDataType*)malloc(sizeof(HPDataType) * hp->capacity);
}

// 堆的销毁
void HeapDestory(Heap* hp)
{
	assert(hp);
	assert(hp->a);
	free(hp->a);
	hp->a = NULL;
}

void swap(HPDataType* a, HPDataType* b)
{
	int tmp = *a;
	*a = *b;
	*b = tmp;
}
void ADjustup(Heap* hp)
{
	int child = hp->size - 1;
	int parent = (child - 1) / 2;
	while (child > 0)
	{
		if (hp->a[child] < hp->a[parent])
		{
			swap(&(hp->a[child]), &(hp->a[parent]));
			child = parent;
			parent = (parent - 1) / 2;
		}
		else
			break;
	}
}
// 堆的插入
void HeapPush(Heap* hp, HPDataType x)
{
	assert(hp);
	assert(hp->a);

	if (hp->size == hp->capacity)
	{
		int newcapacity = hp->capacity == 0 ? 4 : hp->capacity * 2;
		HPDataType* tmp = realloc(hp->a, sizeof(HPDataType) * newcapacity);
		assert(tmp);
		hp->a = tmp;
		hp->capacity = newcapacity;
	}

	hp->a[hp->size] = x;
	hp->size++;
	ADjustup(hp);
}

void ADjustdown(Heap* hp)
{
	int parent = 0;
	int child = parent * 2 + 1;
	while (child<hp->size)
	{
	
		
		if (child + 1 < hp->size && hp->a[child + 1] < hp->a[child])
		{
			child = child + 1;
		}
		if (hp->a[child] < hp->a[parent])
		{
			swap(&hp->a[child], &hp->a[parent]);
			parent = child;
			child = parent * 2 + 1;
		}
		else
			break;

	}
}
// 堆的删除
void HeapPop(Heap* hp)
{
	assert(hp);
	assert(hp->size > 0);
	swap(&(hp->a[0]), &(hp->a[hp->size - 1]));
	hp->size--;
	ADjustdown(hp);
}

// 取堆顶的数据
HPDataType HeapTop(Heap* hp)
{
	assert(hp);
	assert(hp->size>0);
	return hp->a[0];
}

// 堆的数据个数
int HeapSize(Heap* hp)
{
	assert(hp);
	return hp->size;
}

// 堆的判空
bool HeapEmpty(Heap* hp)
{
	assert(hp);
	return hp->size == 0;
}

本篇文章就到这里啦,下期我们与链树相会! 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C#的checked关键字判断是否溢出

目录 一、定义 二、示例&#xff1a; 三、生成&#xff1a; 一、定义 使用checked关键字处理溢出。 在进行数学运算时&#xff0c;由于变量类型不同&#xff0c;数值的值域也有所不同。如果变量中的数值超出了变量的值域&#xff0c;则会出现溢出情况&#xff0c;出现溢出…

自然语言处理3——玩转文本分类 - Python NLP高级应用

目录 写在开头1. 文本分类的背后原理和应用场景1.1 文本分类的原理1.2 文本分类的应用场景 2. 使用机器学习模型进行文本分类&#xff08;朴素贝叶斯、支持向量机等&#xff09;2.1 朴素贝叶斯2.1.1 基本原理2.1.2 数学公式2.1.3 一般步骤2.1.4 简单python代码实现 2.2 支持向量…

【惊喜揭秘】xilinx 7系列FPGA时钟区域内部结构大揭秘,让你轻松掌握!

本文对xilinx 7系列FPGA的时钟路由资源进行讲解&#xff0c;内容是对ug472手册的解读和总结&#xff0c;需要该手册的可以直接在xilinx官网获取&#xff0c;或者在公众号回复“xilinx手册”即可获取。 1、概括 7系列器件根据芯片大小不同&#xff0c;会有8至24个时钟区域&…

在Pyqt5的QtWidgets.QGraphicsView上绑定matplotlib.figure实现绘图

matplotlib的基础类figure相当于一个View窗口类&#xff08;实际上&#xff0c;每一个figure是由更底层canvas来控制的&#xff0c;大概有点类似CAD的layers层的概念&#xff09;&#xff0c;是一个可绘制显示图形的View区域&#xff0c;也称画布&#xff08;figure&#xff09…

OSG绘制视锥体(升级版)

OSG绘制视锥体&#xff0c;这一篇增加设置相机参数接口&#xff0c;支持通过eye、center、up设置相机参数。 代码如下&#xff1a; #include "stdafx.h" #include <osgViewer/Viewer> #include <osg/ShapeDrawable> #include <osg/Geode> #includ…

阿里开源大模型 Qwen-72B 私有化部署

近期大家都知道阿里推出了自己的开源的大模型千问72B&#xff0c;据说对于中文非常友好&#xff0c;在开源模型里面&#xff0c;可谓是名列前茅。 千问拥有有强大的基础语言模型&#xff0c;已经针对多达 3 万亿个 token 的多语言数据进行了稳定的预训练&#xff0c;覆盖领域、…

最新Redis7哨兵模式(保姆级教学)

一定一定要把云服务器的防火墙打开一定要&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;否则不成功&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&…

C++继承与派生——(7)类型兼容原则

归纳编程学习的感悟&#xff0c; 记录奋斗路上的点滴&#xff0c; 希望能帮到一样刻苦的你&#xff01; 如有不足欢迎指正&#xff01; 共同学习交流&#xff01; &#x1f30e;欢迎各位→点赞 &#x1f44d; 收藏⭐ 留言​&#x1f4dd; 站在巨人的肩上&#xff0c;是为了超过…

lv13 内核模块参数和依赖

1 模块传参 1.1 模块参数设置 将指定的全局变量设置成模块参数 module_param(name,type,perm);//将指定的全局变量设置成模块参数 /* name:全局变量名 type&#xff1a;使用符号 实际类型 传参方式bool bool insmod xxx.ko 变量名0 …

CorelDRAW 2024最新版本功能介绍含注册机序列号

始于1989年并不断推陈出新,致力为设计工作者提供更高效的设计工具&#xff01;CorelDRAW滋养并见证了一代设计师的成长&#xff01;在最短的时间内交付作品&#xff0c;CorelDRAW的智能高效会让你一见钟情&#xff01;CorelDRAW 全称“CorelDRAW Graphics Suite“&#xff0c;也…

云原生|kubernetes|kubernetes资源备份和集群迁移神器velero的部署和使用

前言&#xff1a; kubernetes集群需要灾备吗&#xff1f;kubernetes需要迁移吗&#xff1f; 答案肯定是需要的 那么&#xff0c;如何做kubernetes灾备和迁移呢&#xff1f;当然了&#xff0c;有很多的方法&#xff0c;例如&#xff0c;自己编写shell脚本&#xff0c;或者使用…

【科研】[3.番外篇] 常见基础科研词汇的介绍!非常基础的一期,大家选择观看哟~

文章目录 我的总结&#xff1a; 我的总结&#xff1a; 轻松一刻 视频来源&#xff1a;https://www.bilibili.com/video/BV1ca4y1k7AK 本期是一期插播视频&#xff0c;内容非常基础&#xff0c;面向超级新手的词汇讲解&#xff1a; Baseline: 基线&#xff0c;文章中用于比较…

io流——转换流

1 为什么需要转换流 为啥 乱码的原理 怎么解决&#xff1f; 1 让两边编码一样 2 使用转换流 2 转换流

大数定律中心极限定理

1.切比雪夫不等式 切比雪夫不等式可以对随机变量偏离期望值的概率做出估计&#xff0c;这是大数定律的推理基础。以下介绍一个对切比雪夫不等式的直观证明。 1.1 示性函数 对于随机事件A&#xff0c;我们引入一个示性函数 I A { 1 , A发生 0 , A不发生 I_A\begin{cases} 1&…

西门子PLC通过PROFINET协议与多功能电表通讯

西门子PLC通过PROFINET协议与多功能电表通讯 项目要求 西门子S71200PLC需要通过PROFINET协议和多功能电表通讯&#xff0c;读取线电压、相电压、线电流、相电流、有功功率、无功功率等参数。 项目实施 采用网关NET90-PN-MBT&#xff08;以下简称“网关”&#xff09;&#…

小米汽车 SU7 技术发布会-智能驾驶猜想,真的“吊打”特斯拉?

核心主题 本人AI数据工程师&#xff0c;看完小米汽车 SU7 技术发布会&#xff0c;主谈智能驾驶猜想。 小米汽车披露&#xff1a;智能驾驶要2024年跻身第一梯队 发布会前沿致敬经典&#xff0c;挺好的毕竟礼多人不怪。 见面道辛苦&#xff0c;必定是江湖。 见面致经典&#…

系列六、Consul

一、Consul 1.1、概述 Consul是一套开源的分布式服务发现和配置管理系统&#xff0c;由HashiCorp公司用Go语言开发。他提供了微服务系统中的服务治理、配置中心、控制总线等功能。这些功能中的每一个功能都可以单独使用&#xff0c;也可以一起使用以构建全方位的服务网格&…

基于 Vue3 和 WebSocket 实现的简单网页聊天应用

首先附上项目介绍,后面详细解释技术细节 1. chat-websocket 一个基于Vue3和WebSocket的简易网络聊天室项目&#xff0c;包括服务端和客户端部分。 项目地址 websocket-chat 下面是项目的主要组成部分和功能&#xff1a; 项目结构 chat-websocket/ |-- server/ # WebSocket 服…

解锁大数据世界的钥匙——Hadoop HDFS安装与使用指南

目录 1、前言 2、Hadoop HDFS简介 3、Hadoop HDFS安装与配置 4、Hadoop HDFS使用 5、结语 1、前言 大数据存储与处理是当今数据科学领域中最重要的任务之一。随着互联网的迅速发展和数据量的爆炸性增长&#xff0c;传统的数据存储和处理方式已经无法满足日益增长的需求。…

C++进阶--二叉树进阶(二叉搜索树)

二叉树进阶&#xff08;二叉搜索树&#xff09; 一、二叉搜索树1.1 二叉搜索树的概念 二、二叉搜索树的结构2.1 结点结构2.2 树结构 三、二叉搜索树的操作&#xff08;非递归&#xff09;3.1 二叉搜索树的插入3.2 二叉搜索树的查找3.3 二叉搜索树的中序遍历3.4 二叉搜索树的删除…