数据结构-二叉树·堆(顺序结构的实现)

news2024/11/27 14:29:58

🎉个人名片:

🐼作者简介:一名乐于分享在学习道路上收获的大二在校生
🐻‍❄个人主页🎉:GOTXX
🐼个人WeChat
ILXOXVJE
🐼本文由GOTXX原创,首发CSDN🎉🎉🎉
🕊系列专栏:零基础学习C语言----- 数据结构的学习之路

🐓每日一句:如果没有特别幸运,那就请特别努力!🎉🎉🎉
————————————————

🎉文章简介:

本篇文章对    树的相关概念及结构,二叉树(堆)的概念及结构,二叉树顺序结构及实现的相关知识详细讲解!二叉树链式结构 在下一章讲解!

如果您觉得文章不错,期待你的一键三连哦,你的鼓励是我创作动力的源泉,让我们一起加油,一起奔跑,让我们顶峰相见!!!🎉🎉🎉


目录

一.树的概念及结构

1.1树的概念

相关概念:

1.2树的表示

二.二叉树的概念及结构

2.1二叉树的概念

二叉树:

2.2两个特殊的二叉树

满二叉树:

完全二叉树:

三.二叉树顺序结构及实现

3.1二叉树顺序结构

堆在存储的分类:大根堆,小根堆

3.2二叉树(堆)顺序结构的实现

这里重点分析向上/向下调整的函数

向上调整:

向下调整:

完整代码:Heap.h    Heap.c     


一.树的概念及结构

1.1树的概念

图一
图一
图二
图二

树是一种非线性的数据结构,它是由k个节点(k>=0)组成的具有层次关系的一个集合,如图一所示,把上图倒过来,如图二所示,看起来像一棵树,所以被叫作树;

类似于树的特点,把最上面的那个结点(A)叫作根结点

除了根结点,其余的结点又可以分为若干个类似于树的子树,如下图:

所以树是递归定义的;

相关概念:

1.结点的度:及该结点含有子树的个数(有几个孩子),如上图:1的度为3,2的度为1,4的度为2;

2.叶结点(终端结点):度为0的结点,如上图的3,5,6,7

3.分枝结点(非终端结点):根结点与叶结点以外的结点,如2,4

4.双亲结点(父结点):一个结点含有子结点,该结点称为子结点的父结点,如1是2,3,4的父结点,4是6,7的父结点;

5.孩子结点(子结点):如5是2的子结点,4是1的子结点;

6.兄弟结点:有相同父结点的结点称为兄弟结点,如6,7的父结点都是4,所以6,7是兄弟结点;

7.树的度:一棵树中,最大的结点的度称为树的度,如上面的树的度是3(因为1的度最大,为3);

8.结点的层次:根为第一层,往下一次类推;

9.树的高度(深度):如上图,树的高度为3;

10.森林:有许多互不相交的树组成的集合;

1.2树的表示

最常见的是孩子兄弟表示法

双亲表示法(一般使用结构体数组):只存储双亲的下标或指针;

例如:

上面这个树用双亲表示法表示:

蓝色存储的该结点的父结点的下标或指针;

没有父亲就存储-1(-1不是个有效的下标);

二.二叉树的概念及结构

2.1二叉树的概念

二叉树:

1.不存在度大于2的结点的树;最多两个,可以是1个或则0个;

度为0(空树);

2.二叉树的子树 有左右子树之分,次序不能颠倒,所以二叉树是有序的;

2.2两个特殊的二叉树

满二叉树:

一个二叉树,如果每一层的结点数都达到最大值,这个数就是满二叉树;

假设一个满二叉树有h层,则该二叉树的总的结点为2^h-1;

完全二叉树:

是一个深度为k的有n个节点的二叉树,对树中的节点按从上至下、从左到右的顺序进行编号,如果编号为i1≤i≤n的结点与满二叉树中编号为i的结点在二叉树中的位置相同;

三.二叉树顺序结构及实现

3.1二叉树顺序结构

根据完全二叉树的特点,可以得出这样的结论:

如果完全二叉树用数组存储,那么可以得到任意一个父结点,可以通过下标找到孩子,通过孩子下标也可以找到父结点的下标;

规律如下:

liftchild = perent*2+1;

rightchild = parent*2+2;

parent = (child-1)/2;

堆在存储的分类:大根堆,小根堆

3.2二叉树(堆)顺序结构的实现

这里重点分析向上/向下调整的函数

向上调整:

思想:将插入的数据尾插到数组里面,根据父结点与孩子结点下标的关系向上比较做调整,如果父亲结点的数据大于(小于)孩子结点,就交换:如图:

实现代码:

//交换函数
void Swap(HPDataType* x, HPDataType* y)
{
	HPDataType tmp = *x;
	*x = *y;
	*y = tmp;
}

//向上调整
void Adjustup(HPDataType* a, int child)
{
	assert(a);

	int parent = (child - 1) / 2;
	while (child>0)
	{
		if (a[parent] > a[child])
		{
			Swap(&a[parent], &a[child]);
			child = parent;
			parent = (parent - 1) / 2;
		}
		else
		{
			break;
		}
	}
}

向下调整:

思想:如果我们要删除堆顶(根)的结点,如果直接删除,然后向前覆盖,堆的顺序就会改变,不再是大堆(小堆),如图,这里就需要用到向下调整,先将最后一个数据与第一个数据交换,再将最后一个数据删除,这样保证了除了根,下面的结点都是大堆(小堆);

然后再用根和两个孩子中较小的一个交换,一次向下重复以上动作,图解如下:

实现代码:

//向下调整
void Adjustdown(HPDataType* a, int parent,int n)
{
	assert(a);
	int child = parent * 2 + 1;
	while (child<n)
	{
		//假设左孩子小
		if (child+1<n && a[child] > a[child + 1])   //假设错误,修正
		{
			child = child + 1;
		}
		if (a[child] < a[parent])
		{
			Swap(&a[parent], &a[child]);
			parent = child;
			child = child * 2 + 1;
		}
		else
		{
			break;
		}
	}
}

完整代码:Heap.h    Heap.c     

Heap.h

#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>
#include<string.h>

#define HPDataType int


typedef struct Heap
{
	//存储数据的数组
	HPDataType* a;
	int size;
	int capacity;
}Heap;

//初始化函数,两种
//先不开空间,使用的时候再开
void HeapInit(Heap* php);
//已经有一个数组的数据,先开空间,把一个数组的数据放到堆数组里面
void HeapInitArray(Heap* php,int* a,int n);

//摧毁函数,防止内存泄露
void HeapDestory(Heap* php);
//打印函数
void HeapPrintf(Heap* php);

//向上调整函数
void Adjustup(HPDataType* a, int child);
//向下调整函数
void Adjustdown(HPDataType* a, int child,int n);

//向堆里面插入数据的函数
void HeapPush(Heap* php, HPDataType x);
//把堆里面的根结点Pop出去的函数
void HeapPop(Heap* php);

//取出根结点数据的函数
HPDataType HeapTop(Heap* php);

//判断堆是否为空的函数
bool HeapEmpty(Heap* php);

Heap.c

#define _CRT_SECURE_NO_WARNINGS 1
#include "Heap.h"

void HeapInit(Heap* php)
{
	assert(php);

	php->a = NULL;
	php->capacity = 0;
	php->size = 0;

}

void HeapInitArray(Heap* php,int* a,int n)
{
	assert(a);
	assert(php);

	php->a = (HPDataType*)malloc( n * sizeof(int));
	if (php->a == NULL)
	{
		perror("malloc fail");
		exit(-1);
	}
	memcpy(php->a, a, n * sizeof(int));

	//向上调整建堆
	for (int i = 1; i < n; i++)
	{
		Adjustup(php->a, i);
	}

	php->size = n;
	php->capacity = n;
}

void HeapDestory(Heap* php)
{
	assert(php);

	php->a = NULL;
	php->capacity = php->size = 0;

}

void HeapPrintf(Heap* php)
{
	assert(php);

	for (int i = 0; i < php->size; i++)
	{
		printf("%d ",php->a[i]);
	}
	printf("\n");
}

//交换函数
void Swap(HPDataType* x, HPDataType* y)
{
	HPDataType tmp = *x;
	*x = *y;
	*y = tmp;
}

//向上调整函数
void Adjustup(HPDataType* a, int child)
{
	assert(a);

	int parent = (child - 1) / 2;
	while (child>0)
	{
		if (a[parent] > a[child])
		{
			Swap(&a[parent], &a[child]);
			child = parent;
			parent = (parent - 1) / 2;
		}
		else
		{
			break;
		}
	}
}

//向下调整函数
void Adjustdown(HPDataType* a, int parent,int n)
{
	assert(a);
	int child = parent * 2 + 1;
	while (child<n)
	{
		//假设左孩子小
		if (child+1<n && a[child] > a[child + 1])
		{
			child = child + 1;
		}
		if (a[child] < a[parent])
		{
			Swap(&a[parent], &a[child]);
			parent = child;
			child = child * 2 + 1;
		}
		else
		{
			break;
		}
	}
}

//插入函数
void HeapPush(Heap* php, HPDataType x)
{
	assert(php);

	if (php->capacity == php->size)
	{
		int newcapacity = php->capacity == 0 ? 4 : 2 * php->capacity;
		HPDataType* tmp = (HPDataType*)realloc(php->a, newcapacity * sizeof(HPDataType));
		if (tmp == NULL)
		{
			perror("realloc fail");
			exit(-1);
		}
		php->a=tmp;
		php->capacity = newcapacity;
	}
	php->a[php->size] = x;
	php->size++;

	Adjustup(php->a,php->size-1);
}

//删除堆顶结点
void HeapPop(Heap* php)
{
	assert(php);

	Swap(&php->a[0], &php->a[php->size - 1]);
	php->size--;

	Adjustdown(php->a, 0,php->size);

}

//取出堆顶数据的函数
HPDataType HeapTop(Heap* php)
{
	assert(php);

	return php->a[0];
}

//判空函数
bool HeapEmpty(Heap* php)
{
	assert(php);

	return php->size;
}

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

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

相关文章

游戏推荐《塞尔达传说王国之泪》

塞尔达传说&#xff1a;王国之泪 播报编辑讨论12上传视频 2023年任天堂企划制作本部开发的动作冒险游戏 《塞尔达传说 王国之泪》是任天堂企划制作本部开发的动作冒险游戏&#xff0c;为《塞尔达传说》主系列的第20作、《塞尔达传说&#xff1a;旷野之息》的正统续篇&#xf…

块级作用域的理解

块级作用于的概念 由一对花括号{}中的语句集都属于一个块&#xff0c;在这个{}里面包含的块内定义的所有变量在代码块外都是不可见的&#xff0c;因此称为块级作用域。 作用域永远都是任何一门语言的重中之中&#xff0c;因为它控制着变量和参数的可见性和生命周期。讲到这里&…

uniapp打包安卓app后获取签名证书的SHA1,SHA256,MD5等信息

获取签名证书信息的方法有两种 1.dcloud开发者中心生成证书查看 在证书详情中可以查看&#xff0c;还可以下载证书 2.使用命令查看 1.先安装jre8&#xff0c;再配置一下环境变量 jre8下载地址 2.将打好的正式版app 后缀改为 .zip&#xff0c;解压 3.打开META-INF目录&…

基于卷积神经网络的抗压强度预测,基于卷积神经网络的抗折强度预测

目录 背影 卷积神经网络CNN的原理 卷积神经网络CNN的定义 卷积神经网络CNN的神经元 卷积神经网络CNN的激活函数 卷积神经网络CNN的传递函数 卷积神经网络CNN抗压强度预测 完整代码:基于卷积神经网络的抗压强度和抗折强度预测,基于CNN的抗压强度和抗折强度预测(代码完整,数据…

搭建第一个区块链网络与一键部署WeBASE步骤

官网 搭建第一个区块链网络 — FISCO BCOS v2 v2.9.0 文档 (fisco-bcos-documentation.readthedocs.io) 一键部署 — WeBASE v1.5.5 文档 (webasedoc.readthedocs.io) 步骤 默认如MySQL、Python、java等依赖已经引入 1.创建操作目录, 下载安装脚本 创建操作目录 cd ~ &a…

Hydra post登录框爆破

文章目录 无token时的Hydra post登录框爆破带Token时的Hydra post登录框爆破 无token时的Hydra post登录框爆破 登录一个无验证码和token的页面&#xff0c;同时抓包拦截 取出发送数据包&#xff1a;usernameadb&password133&submitLogin 将用户名和密码替换 userna…

安科瑞变电站综合自动化系统在青岛海洋科技园应用

安科瑞 耿敏花 摘 要&#xff1a;变电站综合自动化系统是将变电站内的二次设备经过功能的组合和优化设计&#xff0c;利用先进的计算机技术、通信技术、信号处理技术&#xff0c;实现对全变电站的主要设备和输、配电线路的自动监视、测量、控制、保护、并与上级调度通信的综合性…

教你怎么用Python每天自动给女朋友免费发短信

今天的教程就是教大家怎么发送免费短信给女朋友。 发送短信接口&#xff0c;我知道的常见的有两个平台&#xff0c;一个是 twilio&#xff0c;可以免费发短信 500 条&#xff0c;可发任意信息&#xff0c;一个是腾讯云&#xff0c;可以免费发短信 100 条&#xff0c;需要申请短…

【C语言初阶】函数

目录 一、函数是什么 二、C语言中函数的分类 2.1 库函数&#xff1a; strcpy memset 2.2 自定义函数 三、函数的参数 3.1 实际参数&#xff08;实参&#xff09; 3.2 形式参数&#xff08;形参&#xff09; 四、函数的调用 4.1 传值调用 4.2 传址调用 五、函数的嵌…

【广州华锐互动】VR历史古城复原:沉浸式体验古代建筑,感受千年风华!

在科技日新月异的今天&#xff0c;虚拟现实&#xff08;VR&#xff09;技术已经成为了我们生活中不可或缺的一部分。从娱乐游戏到医疗健康&#xff0c;从教育培训到房地产销售&#xff0c;VR技术的应用领域日益广泛。而近年来&#xff0c;VR技术在文化遗产保护和古迹复原方面的…

警惕Mallox勒索病毒的最新变种mallox,您需要知道的预防和恢复方法。

尊敬的读者&#xff1a; 在这个数字时代&#xff0c;恶意软件不再是仅限于技术领域的威胁&#xff0c;而是每个人都可能面临的潜在风险。其中&#xff0c;.mallox勒索病毒崭露头角&#xff0c;它不仅能够以不可思议的方式加密您的数据&#xff0c;还能要求您支付赎金以获取解密…

行业观察:数字化企业需要什么样的数据中心

伴随着数字经济在中国乃至全球的高速发展&#xff0c;数字化转型已经成为广大企业的必经之路。而作为数字经济的核心基础设施&#xff0c;数据中心充当了接收、处理、存储与转发数据流的“中枢大脑”&#xff0c;对驱动数字经济发展和企业数字化转型起到了极为关键的重要作用。…

vue中的rules表单校验规则使用方法 :rules=“rules“

一、el-form里面必写属性值 :ref"dataForm" // 提交表单时进行校验 :rules"rules" // return 下的校验规则 :model"userForm" // 绑定表单的值 <el-formref"dataForm" // 必写属性值:rules"rules"…

通过在Z平面放置零极点的来设计数字滤波器

文章来源地址&#xff1a;https://www.yii666.com/blog/393376.html 通过在Z平面放置零极点的来设计数字滤波器 要求&#xff1a;设计一款高通滤波器&#xff0c;用在音频信号处理过程中&#xff0c;滤掉100Hz以下的信号。 实现方法&#xff1a;通过在Z平面放置零极点的来设…

(景行锐创) 高性能计算平台 Pytorch 深度学习环境超详细教程

文章目录 前言1. 账号申请2. 登录高算平台网站3. 安装 Xshell&#xff0c;Xftp 软件4. 连接高算平台5. 安装 Anaconda6. 安装 CUDA7. 配置 cuDNN8. 安装 torch 和 torchvision9. 提交作业测试10. 解压与压缩文件11. 其他结语 前言 目前一些学校为了便于师生进行大规模的计算任…

Aop自定义注解生成日志

Aop自定义注解生成日志 1.编写自定义注解 //表示此注解可以标注在方法上 Target(ElementType.METHOD) //运行时生效 Retention(RetentionPolicy.RUNTIME) public interface OpetionLog {//定义一个变量&#xff0c;可以接收参数String value() default "";}2.Cont…

Leetcode 剑指 Offer II 051. 二叉树中的最大路径和

题目难度: 困难 原题链接 今天继续更新 Leetcode 的剑指 Offer&#xff08;专项突击版&#xff09;系列, 大家在公众号 算法精选 里回复 剑指offer2 就能看到该系列当前连载的所有文章了, 记得关注哦~ 题目描述 路径 被定义为一条从树中任意节点出发&#xff0c;沿父节点-子节…

微信CRM系统在电商行业的应用

随着移动互联网的快速发展&#xff0c;微信已经成为电商行业重要的营销和客户服务渠道。然而&#xff0c;如何在海量的微信消息中保持客户沟通、提升客户体验、实现精准营销&#xff0c;是电商行业面临的一大挑战。微信CRM系统作为一种新型的客户关系管理工具&#xff0c;可以有…

【qemu逃逸】HWS2017-FastCP

前言 虚拟机用户名&#xff1a;root 虚拟机密码&#xff1a;无密码 本题有符号&#xff0c;所以对于设备定位啥的就不多说了&#xff0c;直接逆向设备吧。 设备逆向 在 realize 函数中设置一个时钟任务&#xff0c;并且可以看到只注册了 mmio&#xff0c;大小为 0x100000。…

Web应用多账号系统设计及微信扫码登录实现

1 前言概述 公司对功能测试&#xff0c;性能测试&#xff0c;安全测试等等都做了比较好的自动化后&#xff0c;急需要一个MIS系统来统一管理这些结果及报表。 此MIS系统特点如下&#xff1a; 仅内部人员使用部署在公网 基于如上特点&#xff0c;显然让公司的人为这样一个…