【数据结构】二叉树-堆(上)

news2024/11/17 7:38:39

在这里插入图片描述
个人主页~


二叉树-堆

  • 一、树的概念及结构
    • 1、概念
    • 2、相关概念
    • 3、树的表示
    • 4、树的实际应用
  • 二、二叉树的概念和结构
    • 1、概念
    • 2、特殊二叉树
    • 3、二叉树的性质
    • 4、二叉树的存储结构
      • (1)顺序存储
      • (2)链式存储
  • 三、二叉树的顺序结构以及实现
    • 1、二叉树的顺序结构
    • 2、堆的概念及结构
      • (1)小根堆
      • (2)大根堆
    • 3、堆的实现
      • (1)堆的向上调整算法--堆的创建
        • ①一般方法
        • ②向上调整建堆
      • (2)向上调整算法的时间复杂度
      • (3)向下调整算法维护堆
      • (4)向下调整算法的时间复杂度

一、树的概念及结构

在我们学习二叉树之前,我们先要了解一下树的概念,二叉树就是一种树
在这里插入图片描述

1、概念

树是一种非线性的数据结构,它是由n个有限节点组成一个具有层次关系的集合,因为根据它所画出的抽象图看起来像一棵倒挂着的树,它的根朝上,树叶朝下

一棵树最顶上的节点叫做根节点,一棵树有且只有一个根节点,根节点没有前驱节点也就是说根节点上面就没有节点了

除了根节点以外,其余节点被分成N个互不相交的集合,我们形象的来说,就是一棵树的叶子和树枝是多对一的概念,也就是说一个树枝可以有多个叶子或者没有叶子,但是一个叶子只能长在一个树枝上,一条小树枝只能长在一条大树枝上,所以树是递归定义的

2、相关概念

节点的度:一个节点含有子树的个数(A的度是3,C的度是0)

叶节点(终端节点):度为0的节点称为叶节点(GHIJKL)

分支节点(非终端节点):度不为0的节点(ABDF)

子节点:一个节点含有的子树的根结点(B是A的子节点)

父节点:若一个节点含有子节点,则这个节点称为其子节点的父节点(A是B的父节点)

兄弟节点:这里的兄弟指的是亲兄弟,也就是具有相同父节点的节点(BCD是兄弟节点)

树的度:整棵树的度是这棵树中的节点的度中最大的那个度(这棵树最大是3)

节点的层次:根为第一层,根的子节点为第二层,子节点的子节点为第
三层,以此类推(第一层:A;第二层:BCD;第三层:FGHI;第四层:JKL)

树的高度(深度):树中节点的最大层次(四层)

堂兄弟节点:父亲为同一层的节点(HI)

节点的祖先:从根到该节点所经分支上的所有节点(J节点的祖先是ABF)

子孙:以某节点为根的子树中任意节点都称为该节点的子孙(B-L都是A的子孙)

森林:由N棵(N>0)互不相交的树组成的集合称为森林

树的概念都是由人类的亲缘关系决定的,我们在记忆的时候可以结合我们人类的亲缘关系来记忆

3、树的表示

树的表示方法有很多种,如果我们再像以前一样定义一个结构体,其中存放指针和数据,那样就不行了,因为我们不知道一个节点有多少子树,这样就没办法定义树的节点的结构体,这里,我们有一个最好的办法就是左孩子右兄弟法

左孩子右兄弟法:

typedef int DataType;
struct Node
{
    struct Node* _firstChild1; // 第一个孩子结点,也就是最左边的孩子
    struct Node* _pNextBrother; 
    // 指向其下一个兄弟结点,就是右边第一个兄弟
    DataType _data; // 结点中的数据域
};

左孩子右兄弟法就是一个指针指向左边第一个孩子,右指针指向右边第一个兄弟
在这里插入图片描述
图画的不太好看,将就一下
红色的线是左孩子
蓝色的线是右兄弟
这样我们可以简洁并且快速地找到这棵树所有的分支

4、树的实际应用

文件系统的目录就是树的应用
在这里插入图片描述
E盘:
在这里插入图片描述
这里就是树的应用,文件系统的目录是用树的结构实现的

二、二叉树的概念和结构

1、概念

二叉树就是在树的基础上加上特殊
二叉树是由一个根节点加上一个左子树和一个右子树组成的
二叉树不存在度大于2的节点
二叉树是有序树,因为它的子树有左右之分,次序不能颠倒

2、特殊二叉树

(1)满二叉树
一个二叉树,如果每一个层的节点数都达到最大值,那么这个二叉树就是满二叉树
在这里插入图片描述

(2)完全二叉树
完全二叉树是效率很高的二叉树,它的最后一层可以不满,最后一层之前的层都是满的,然后最后一层的节点是需要按序排列的,满二叉树是一种特殊的完全二叉树
在这里插入图片描述

3、二叉树的性质

若规定根节点的层数为1,具有n个节点的满二叉树的深度h=log₂(n+1)

对于具有n个节点的完全二叉树,如果按照从上到下,从左到右的数组顺序对所有节点从0开始编号则对于序号为i的节点有如下几个性质:
①若i>0,i位置节点的父亲序号:(i-1)/2;i=0,i为根节点编号,无父亲节点
②若2i+1<n,左孩子序号为2i+1并且2i+1≥n,否则无左孩子
③若2i+2<n,右孩子序号为2i+2并且2i+2≥n,否则无右孩子

4、二叉树的存储结构

二叉树有两种存储结构,一种是顺序存储,另一种是链式存储

(1)顺序存储

顺序存储就是使用数组来存储,一般只适合表示完全二叉树,因为不是完全二叉树会有空间上的浪费,二叉树顺序存储在物理上是一个数组,在逻辑上是一颗二叉树

(2)链式存储

链式存储就是使用链表来存储,通常的方法是链表节点由三个域组成,分别是数据域以及左右指针域,左右指针存储左右孩子的地址

链式结构又分为二叉链和三叉链,这里使用的是二叉链

typedef int BTDataType;
// 二叉链
struct BinaryTreeNode
{
	struct BinTreeNode* Left; // 指向左孩子
	struct BinTreeNode* Right; // 指向右孩子
	BTDataType data; // 值域
};

// 三叉链
struct BinaryTreeNode
{
	struct BinTreeNode* Parent; // 指向父节点
	struct BinTreeNode* Left; // 指向左孩子
	struct BinTreeNode* Right; // 指向右孩子
	BTDataType data; // 值域
};

三、二叉树的顺序结构以及实现

1、二叉树的顺序结构

现实中我们经常把堆(一种二叉树)使用顺序结构的数组来存储,这里的堆与malloc位置的堆的概念是不同的,malloc位置的堆是操作系统中的内存管理,这里的堆是我们人为实现的一种数据结构

2、堆的概念及结构

把一堆数据按照完全二叉树的顺序存储方式存储在一个一维数组中,并且满足第i项第2i+2项,i为自然数,则称为堆,根节点最大的堆叫大堆(大根堆),根节点最小的堆叫小堆(小根堆)

性质:
①堆总是一颗完全二叉树
②堆中某个节点的值总是不大于或不小于其父节点的值

(1)小根堆

逻辑结构:
在这里插入图片描述
物理结构(存储结构):
在这里插入图片描述

(2)大根堆

逻辑结构:

在这里插入图片描述
物理结构(存储结构):
在这里插入图片描述
这里的存储结构中的数据不一定是有序的,也可以不是升序或者降序,但是大堆的父节点一定比子节点大,小堆的父节点一定比子节点小

3、堆的实现

(1)堆的向上调整算法–堆的创建

①一般方法

我们在使用堆的向下调整算法之前要保证左右子树都要是堆,那么在使用之前我们先要创建堆

我们创建一个数组,在逻辑上可以看做一颗完全二叉树,然后我们通过算法把它构建成为一个堆,从倒数第一个叶子节点开始调整一直到根节点,就可以调整成堆

int arr[] = {1,4,7,2,5,9};

在这里插入图片描述
最后的9与它的父节点7交换:
在这里插入图片描述
1<9再交换
在这里插入图片描述
然后再检查最后一个叶子节点,重复上面的步骤,虽然这样最终可以建立一个堆,但这样效率特别低,所以我们有了向上调整算法来建堆

②向上调整建堆

现在我们有一个数组,在逻辑上看成一棵完全二叉树,我们要创建一个堆,可以用向上调整算法

void AdjustUp(HPDataType* a, int child)
{
	int parent = (child - 1) / 2;//因为除法向下取整,所以右孩子也能
	//因为是一颗完全二叉树,所以父节点总是可以通过子节点减1除以二找到
	//while (parent >= 0)
	while (child > 0)//这里用子节点作为循环条件,因为child可能调整到根节点上
	{
		if (a[child] > a[parent])
		{
			Swap(&a[child], &a[parent]);
			child = parent;
			parent = (child - 1) / 2;
		}//大于就交换,把此时的父节点变成子节点,父节点的父节点变成父节点,比较上一层的关系
		else
		{
			break;
		}
	}
}

从二叉树的根节点的左孩子开始调整,按照下标依次调整,最终会建成一个堆
图演示:
在这里插入图片描述
下标1向上调整:
在这里插入图片描述
下标2向上调整:
在这里插入图片描述
下标3调整两次:(第二次小于7,不调)
在这里插入图片描述
下标4调整两次:(第二次小于7,不调)
在这里插入图片描述

下标5调整两次:
第一次:
在这里插入图片描述
第二次:
在这里插入图片描述
这样就建成一个堆了

(2)向上调整算法的时间复杂度

在这里插入图片描述
设树的高度为h
第1层:2^0个节点,需要向上移动0层
第2层:2^1个节点,需要向上移动1层
第3层:2^2个节点,需要向上移动2层
……
第h-1层:2^(h-2)个节点,需要向上移动h-2层
第h层:2^(h-1)个节点,需要向上移动h-1层
将它们相加
在这里插入图片描述
解得原式=2+2^h*(h-2)

遍历一遍为N = 2^h

去掉不重要的项,得时间复杂度O(N*log₂N)

(3)向下调整算法维护堆

当我们需要将堆顶的数据删除掉,那么这个堆就没有了根,如果再重新进行建堆会浪费很多的时间,这里有一种方法的时间复杂度小于重新建堆,这种算法就是向下调整算法

void AdjustDown(int* a, int n, int parent)//n是数组a的数据个数
{
	int child = parent * 2 + 1;//左孩子
	while (child < n)
	{
		// 选出左右孩子中大的那个
		if (child + 1 < n && a[child + 1] > a[child])
		{
			child++;
		}
		if (a[child] > a[parent])
		{
			Swap(&a[parent], &a[child]);
			parent = child;
			child = parent * 2 + 1;
		}//谁大谁是爹
		else
		{
			break;
		}
	}
}

(4)向下调整算法的时间复杂度

在这里插入图片描述
设树的高度为h
第1层:2^0个节点,需要向下移动h-1层
第2层:2^1个节点,需要向下移动h-2层
第3层:2^2个节点,需要向下移动h-3层
……
第h-1层:2^(h-2)个节点,需要向下移动1层
第h层:2^(h-1)个节点,需要向下移动0层
将它们相加之后由错位相减法得
2^h-1-h
因为N = 2^h,所以原式=N-1-log₂N

因为当h趋于无穷大时,N远大于log₂N,所以时间复杂度O(N)


今日分享就到这~
在这里插入图片描述

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

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

相关文章

【Linux-时间管理和内核定时器】

Linux-时间管理和内核定时器 ■ 设置系统节拍率■ 高节拍率和低节拍率的优缺点&#xff1a;■ jiffies 系统节拍数■ get_jiffies_64 这个函数可以获取 jiffies_64 的值■ 处理绕回■ 使用 jiffies 判断超时 ■ jiffies 和 ms、 us、 ns 之间的转换函数在这里插入代码片■ 内核…

pytest:指定测试用例执行顺序

在自动化测试中&#xff0c;测试用例的执行顺序有时对测试结果具有重要影响。本文将介绍如何在pytest框架中使用pytest-ordering插件以及Collection hooks来控制测试用例的执行顺序。 方式1&#xff1a; 使用pytest-ordering插件控制执行顺序 1.1 安装pytest-ordering插件 首…

Android Studio开发之路(十三)主题影响Button颜色问题解决及button自定义样式

一、问题描述 在开发过程中发现安卓的默认主题色是紫色&#xff0c;并且会导致button也是紫色&#xff0c;有时直接在xml布局文件中直接设置button的背景色或者设置背景图片不起效果 方案一、如果是app&#xff0c;可以直接设置主题颜色 比如&#xff0c;将主题设置为白色&a…

【STM32】 独立看门狗配置方法

什么是看门狗 看门狗&#xff08;watchdog&#xff09;指的是一种监控系统或程序&#xff0c;用于定期检测和监控其他系统或程序的运行状态&#xff0c;并在出现问题或故障时采取相应的措施。它可以是硬件设备&#xff0c;也可以是软件程序。 在计算机领域中&#xff0c;看门狗…

Java Swing + MySQL图书借阅管理系统

系列文章目录 Java Swing MySQL 图书管理系统 Java Swing MySQL 图书借阅管理系统 文章目录 系列文章目录前言一、项目展示二、部分代码1.Book2.BookDao3.DBUtil4.BookAddInternalFrame5.Login 三、配置 前言 项目是使用Java swing开发&#xff0c;界面设计比较简洁、适合作…

Taipy快速打造数据驱动的Web应用

Taipy&#xff1a; 用Taipy&#xff0c;让数据洞察与应用开发无缝对接- 精选真开源&#xff0c;释放新价值。 概览 Taipy快速打造数据驱动的 Web 应用。这是一个基于 Python 和 Flask 的项目&#xff0c;结合了 React 等前端技术&#xff0c;为开发者提供了一个简洁、高效的开…

【计算机毕业设计】基于SSM+Vue的新能源汽车在线租赁管理系统【源码+lw+部署文档】

摘 要 随着科学技术的飞速发展&#xff0c;社会的方方面面、各行各业都在努力与现代的先进技术接轨&#xff0c;通过科技手段来提高自身的优势&#xff0c;新能源汽车在线租赁当然也不能排除在外。新能源汽车在线租赁是以实际运用为开发背景&#xff0c;运用软件工程开发方法&…

docker安装Elasticsearch(ES)详细教程

使用Docker来安装Elasticsearch&#xff08;简称ES&#xff09;的详细教程如下&#xff1a; 1. 前提条件 确保你的系统已经安装了Docker&#xff0c;并且Docker服务正在运行。如果你使用的是Windows系统&#xff0c;还需要确保已经启用了Hyper-V和容器功能。 2. 拉取Elastic…

AOP、注解、EL表达、若依权限,Security原理综合分析

AOP、注解、EL表达、若依权限&#xff0c;Security原理综合分析 案例一&#xff1a;更新、创建增强 需求产生 每个表中均有创建时间、创建人、修改时间、修改人等字段。 在操作时候手动赋值&#xff0c;就会导致编码相对冗余、繁琐&#xff0c;那能不能对于这些公共字段在某…

【c++基础】和谐分组

题目描述 s 班共有 n 名学生&#xff0c;按照学号从 1 到的顺序每名学生的身高分别为 a[1],a[2]...a[n]。由于是新学期&#xff0c;s 班需要进行分组&#xff0c;分组的要求如下&#xff1a; 进行分组的组数不能超过 k。 每组的人的学号必须相邻。 由于身高差过大的人分在同一…

今天又一次在金边搬家了

target&#xff1a;离开柬埔寨倒计时-220day 前言 前天&#xff0c;银行的行政告诉我之前住的地方那个中介说房东不租了&#xff0c;重新给我租了个地方&#xff0c;还是在同一栋楼&#xff0c;只是楼层和房间不一样&#xff0c;今天搬了好多躺呀… 周末的日子 昨天下午 同事&…

2024 在Pycharm管理数据库

2024 在Pycharm管理数据库 Pycharm 社区版DataBase Navigator 数据库管理插件(Plugins)安装使用(sqlite为例添加数据) 文章目录 2024 在Pycharm管理数据库一、Pycharm数据库配置1、Database Navigator插件安装2、连接数据库 二、数据库使用1、插件自带基本操作2、控制台操作 …

linux centos循环ping网段ip

循环ping&#xff0c;检测ip是否可用&#xff0c;ping通为正在使用yes&#xff0c;no为不通 vim test.sh#!/bin/bash ip"192.168.1."echo "ping log:" > ./ping.txt for i in {1..128} doping -c 1 -w 1 -W 1 $ip$i | grep -q "ttl" &&a…

Redis崩溃后,如何进行数据恢复的?no.24

本课时我们主要学习通过 RDB、AOF、混合存储等数据持久化方案来解决如何进行数据恢复的问题。 Redis 持久化是一个将内存数据转储到磁盘的过程。Redis 目前支持 RDB、AOF&#xff0c;以及混合存储三种模式。 RDB Redis 的 RDB 持久化是以快照的方式将内存数据存储到磁盘。在…

【数据结构与算法 经典例题】相交链表

&#x1f493; 博客主页&#xff1a;倔强的石头的CSDN主页 &#x1f4dd;Gitee主页&#xff1a;倔强的石头的gitee主页 ⏩ 文章专栏&#xff1a;数据结构与算法刷题系列&#xff08;C语言&#xff09; 期待您的关注 目录 一、问题描述 二、解题思路 方法一:双循环对比法 方…

Minecraft服务器如何搭建

Minecraft这是原版英文名称&#xff0c;在中国大陆被译为《我的世界》&#xff0c;这款游戏很火爆。台湾的很多小伙伴也在玩&#xff0c;其译名为《我的创世神》。现在这款游戏在国内已经被网易代理了。因为这款游戏开源&#xff0c;所以任何人都可以搭建服务器端&#xff0c;如…

群晖nas连接(路由器设置)--群晖配置下文

目录 前言 本文目的与核心 一、打开IPV6和关闭防火墙 路由器后台 二、打开群晖查看是否有ipv6和记住ipv4地址 群晖后台界面 三、路由器设置端口转发 路由器后台 四、打开DDNS-GO的配置页面查看是否配置生效成功 群晖另一个配置后台 五、访问测试 前言 群晖配置上…

Java整合ELK实现日志收集 之 Elasticsearch、Logstash、Kibana

简介 Logstash&#xff1a;用于收集并处理日志&#xff0c;将日志信息存储到Elasticsearch里面 Elasticsearch&#xff1a;用于存储收集到的日志信息 Kibana&#xff1a;通过Web端的可视化界面来查看日志&#xff08;数据可视化&#xff09; Logstash 是免费且开放的服务器端数…

尼姆游戏(人机对战)-Python

1.实验要求 编写程序设计和实现聪明的尼姆游戏&#xff08;人机对战&#xff09;。尼姆游戏是个著名的游戏&#xff0c;有很多变种玩法。两个玩家轮流从一堆物品中拿走一部分。在每一步中&#xff0c;玩家可以自由选择拿走多少物品&#xff0c;但是必须至少拿走一个并且最多只…

保险业务管理系统的设计与实现(论文 + 源码)

保险业务管理系统.zip资源-CSDN文库https://download.csdn.net/download/JW_559/89361419 保险业务管理系统的设计与实现 摘要 历经二十余年的高速发展&#xff0c;我国保险行业的市场竞争已经达到白热化的程度&#xff0c;在同一个城市往往有数十家主体参与保险业务的竞争。保…