当前位置: 首页 > news >正文

堆(数据结构系列11)

目录

前言:

1.优先级队列概念

2.堆的概念

3.堆的存储方式

4.堆的创建

5.创建堆的时间复杂度

6.堆的插入和删除

6.1堆的插入

6.2堆的删除

结束语:


前言:

上一次博客中小编主要与大家分享了 二叉树一些相关的知识点和一些练习题,下面继续跟着小编一起来学习堆的知识吧!本次博客中小编会主要分享一些堆的基础知识。

 

1.优先级队列概念

首先我们先来认识一下什么是优先级队列,我们在前面的博客中提到了队列是一种先进先出的数据结构,但是在有些情况下,操作的数据可能会带有优先级,一般出队列的时候可能会需要优先级高的元素先出队列,那么显然我们的队列是无法做到的,那么在这种情况下,数据结构应该提供两个最基本的操作,一个是返回最高优先级对象,一个是添加新的对象,这种数据结构就是优先级队列。


2.堆的概念

如果有一个关键码的集合K = {k0, k1, k2, ..., kn - 1},把他的所有元素按完全二叉树的顺序存储方式存储在一个一维数组中,并满足:ki <= k2i + 2且ki <= k2i + 2(ki >= k2i + 1 且 ki >= k2i + 2) i = 0,1,2,...,则称为小堆(或者是大堆)。将根结点最大的堆叫做最大堆或大跟堆,根结点最小的堆叫最小堆或小根堆。

如下图所示:
 

小根堆示意图 

大跟堆示意图

堆的性质:

  • 堆中某个结点的值总是不大于或不小于其父结点的值。
  • 堆总是一颗完全二叉树。

 

3.堆的存储方式

从堆的概念我们知道堆是一颗完全二叉树,因此可以层序的规则采用顺序的方式来高效存储。

注意:
对于非完全二叉树,则不适合使用顺序方式进行存储,因此为了能够还原二叉树,空间中必须要存储空结点,就会导致空间利用率比较低。

将元素存储到数组中后,可以根据二叉树章节的性质对树进行还原,假设i为结点在数组中的下标,则有:
如果i为0,则i表示的结点为根节点,否则i结点的双亲结点为(i - 1)/ 2。

如果2 * i + 1小于结点个数,则结点i的左孩子下标为2 * i + 1,否则没有左孩子。

如果2 * i + 2小于结点个数,则结点i的右孩子下标为2 * i + 2,否则没有右孩子。

如上图所示:其中结点的个数为6,那么对于下标为5的结点来说它的父亲结点就是(5 - 1)/ 2。

对于下标为2的结点来说,由于(2 * 2) +  1  <  6,所以她的左孩子就是(2 * 2)+ 1,由于(2  * 2)+ 2 > 6 ,故它没有右孩子。


4.堆的创建

堆的创建是由向下调整来完成的,那么什么是向下调整呢?

向下调整:
我们直接以创建一颗大跟堆来举个例子。

 

这颗二叉树我们应该怎么用向上调整的方式来进行调整为一颗大跟堆或者是小根堆呢?

首先我们下调整都是从二叉树的最后一个结点开始调整的如下图所示:

 

步骤:

1.让parent标记需要调整的结点,child标记parent的左孩子(注意:parent如果有孩子一定先是有左孩子)。

2.如果parent的左孩子存在,即:child  < size ,进以下操作,直到parent的左孩子不存在。

  • parent右孩子是否存在,如果存在那么找到左右孩子中最大的孩子。
  • 让parent与child进行比较,如果parent大于较大的孩子child,调整结束,否则:交换parent与较大的孩子child,交换完之后,parent中小的元素向下移动,可能导致子树不满足对的性质,因此需要继续向下调整,即parent = child,child = parent * 2 + 1;然后继续2。 

注意:

在调整以parent为根的二叉树时,必须要满足parent的左子树和右子树已经是堆了才可以向下调整。 

代码实现:

核心代码:

package 堆;public class Heap {public int[] elem;public int usedSize;public Heap() {this.elem = new int[10];}//初始化堆public void initElem(int[] array) {for (int i = 0; i < array.length; i++) {elem[i] = array[i];usedSize++;}}//创建堆public void createHeap() {//从最后一个结点所对应的父亲结点开始调整//由于最后一个结点的下标值为usedSize - 1,故parent的下标值就是最后一个结点的下标值 - 1再除以2。//即:parent = (usedSize - 1 - 1) / 2//直到调整到parent = 0为止。for (int parent = (usedSize - 1 - 1) / 2; parent >= 0; parent--) {shiftDown(parent,usedSize);}}//向下调整public void shiftDown(int parent, int len) {//知道parent所在的位置之后就先确定要调整的child的结点的位置,首先要调整就是左孩子结点int child = 2 * parent + 1;//至少要有一个左孩子,否则的话就不做调整。while (child < len) {//判断是否存在右孩子,并且右孩子结点大于左孩子结点的值//将child++,保证child指向的左右孩子的最大的那一个if (child + 1 < len && elem[child] < elem[child + 1]) {child++;}//判断最大孩子结点的值与父亲结点的值的大小//如果child所在的位置的值比parent所在位置的值大就将两者互换if (child + 1 < len && elem[child] < elem[child + 1]) {child++;}//判断最大孩子结点的值与父亲结点的值的大小//如果child所在的位置的值比parent所在位置的值大就将两者互换if (elem[child] > elem[parent]) {//互换int tmp = elem[child];elem[child] = elem[parent];elem[parent] = tmp;//换完之后在继续向下调整//让parent指向孩子结点的位置//child指向现在parent所在位置的左孩子结点//继续重复上述的循环,直到不满足条件就说明这课树就已经调整完毕了parent = child;child = parent * 2 + 1;}else {break;}}}
}


测试代码:
 

package 堆;public class Test {public static void main(String[] args) {Heap heap = new Heap();int[] array = {27,15,19,18,28,34,65,49,25,37};heap.initElem(array);heap.createHeap();}
}

结果展示:

创建小根堆和上述代码差不多,只是在比较parent 和child的大小部分与其不同,大家可以下来自己实现一下。 


5.创建堆的时间复杂度

为了简化我们此处使用满二叉树来给大家证明一下。

如下所示:

 

推理如下所示:
 


6.堆的插入和删除

6.1堆的插入

堆的插入需要两步:

  1. 先将元素放入到底层空间中(注意:空间不够需要扩容)
  2. 将最后新插入的结点向上调整,直到满足堆的性质。

向上调整:向上调整就是我们直接拿这个结点和根结点进行比较即可。

示意图如下所示,以大跟堆为例:

核心代码入所示:

//插入一个元素public void offer(int val) {//判满if (isFull()) {//扩容elem = Arrays.copyOf(elem,2 * elem.length);}//将新元素放置在最后一个位置上elem[usedSize++] = val;//向上调整shiftUp(usedSize - 1);}private void shiftUp(int child) {int parent = (child - 1) / 2;while (child > 0) {if (elem[child] > elem[parent]) {//交换int tmp = elem[child];elem[child] = elem[parent];elem[parent] = tmp;//重新赋值child = parent;parent = (child - 1) / 2;}else {break;}}}public boolean isFull() {return usedSize == elem.length;}


结果展示:

6.2堆的删除

堆的删除一定是删除堆顶元素,具体步骤入下:

  1. 将堆顶元素与堆中的最后一个元素互换。
  2. 将堆中有效数据减一
  3. 对堆顶元素进行向下调整

核心代码如下所示:

//删除堆顶元素public void pop() {//1.将堆顶元素与最后一个元素互换int tmp = elem[0];elem[0] = elem[usedSize - 1];elem[usedSize - 1] = tmp;//2.有效长度减一usedSize--;//3.将堆顶元素向下调整shiftDown(elem[0],usedSize);}

结果展示:


结束语:

好啦这节有关于堆的基本知识点小编就与大家分享到这里啦!如果想要继续深入了解的同学继续跟着小编一起走吧!下一次小编将会和大家继续分享一些有关于堆的知识的,希望对大家有所帮助,想要学习的同学记得关注小编和小编一起学习吧!如果文章中有任何错误也欢迎各位大佬及时为小编指点迷津(在此小编先谢过各位大佬啦!)

相关文章:

堆(数据结构系列11)

目录 前言&#xff1a; 1.优先级队列概念 2.堆的概念 3.堆的存储方式 4.堆的创建 5.创建堆的时间复杂度 6.堆的插入和删除 6.1堆的插入 6.2堆的删除 结束语&#xff1a; 前言&#xff1a; 上一次博客中小编主要与大家分享了 二叉树一些相关的知识点和一些练习题&…...

算法训练第四十二天|01背包问题 二维 、01背包问题 一维、416. 分割等和子集

动态规划part0401背包问题 二维01 背包二维dp数组01背包完整c测试代码总结01背包问题 一维一维dp数组&#xff08;滚动数组&#xff09;一维dp01背包完整C测试代码416. 分割等和子集题目描述思路01背包问题总结01背包问题 二维 视频链接&#xff1a;https://www.bilibili.com/…...

Java-如何使用Java将图片和文字拼接在一起(并非是给图片加水印)

之前有遇到一个问题 问题背景&#xff1a;项目中&#xff0c;有一个功能&#xff0c;管理端可以将客户创建的小程序码下载到本地&#xff0c;方便客户将对应门店的小程序码打印出来并张贴到门店&#xff0c;做门店的引流和会员入会。 具体问题&#xff1a;当小程序码的数量较少…...

Metasploit入门到高级【第三章】

来自公粽号&#xff1a;Kali与编程预计更新第一章&#xff1a;Metasploit 简介 Metasploit 是什么Metasploit 的历史和发展Metasploit 的组成部分 第二章&#xff1a;Kali Linux 入门 Kali Linux 简介Kali Linux 安装和配置常用命令和工具介绍 第三章&#xff1a;Metasploi…...

枚举的使用

Java 枚举是一个特殊的类&#xff0c;一般表示一组常量&#xff0c;比如一年的 4 个季节&#xff0c;一个年的 12 个月份&#xff0c;一个星期的 7 天&#xff0c;方向有东南西北等。1 问题如何在类中使用枚举&#xff0c;例如枚举出一年的四个季度&#xff0c;并且通过迭代枚举…...

Python进阶语法

1.1 Python进阶语法 1.1.1 交换变量 一行代码快速交换两个变量&#xff0c;无需创建临时变量。 from icecream import ica 2 b 4 a, b b, a ic(a, b)ic| a: 4, b: 2 1.1.2 链式比较 from icecream import ica 97 if 90 < a < 100:ic(a)ic| a: 97 1.1.3 初始化列表…...

Pyspark_结构化流4

Pyspark 注&#xff1a;大家觉得博客好的话&#xff0c;别忘了点赞收藏呀&#xff0c;本人每周都会更新关于人工智能和大数据相关的内容&#xff0c;内容多为原创&#xff0c;Python Java Scala SQL 代码&#xff0c;CV NLP 推荐系统等&#xff0c;Spark Flink Kafka Hbase Hi…...

Linux cmp 命令

Linux cmp 命令用于比较两个文件是否有差异。 当相互比较的两个文件完全一样时&#xff0c;则该指令不会显示任何信息。若发现有所差异&#xff0c;预设会标示出第一个不同之处的字符和列数编号。若不指定任何文件名称或是所给予的文件名为"-"&#xff0c;则cmp指令…...

Python入门到高级【第五章】

预计更新第一章. Python 简介 Python 简介和历史Python 特点和优势安装 Python 第二章. 变量和数据类型 变量和标识符基本数据类型&#xff1a;数字、字符串、布尔值等字符串操作列表、元组和字典 第三章. 控制语句和函数 分支结构&#xff1a;if/else 语句循环结构&#…...

C语言中(i++)+ (i++)真的每次都等于3吗?

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录前言结论证明首先&#xff0c;登场的是我们的VC6.0&#xff08;还有Linux&#xff09;最后一位&#xff0c;我使用了小熊猫C&#xff08;还有Clion&#xff09;请添加…...

Cursor,程序员的 AI 代码编辑助手

相信大家都或多或少地听说过、了解过 chatGPT &#xff0c;半个月前发布的 GPT-4 &#xff0c;可谓是 AI 赛道上的一个王炸 那么今天咸鱼给大家分享一个开源的 AI 代码编辑器——Cursor&#xff0c;让各位程序员在编程之路上一骑绝尘 &#x1f603; 介绍 Cursor 是一个人工智…...

基于XML的自动装配~

基于XML的自动装配之场景模拟&#xff1a; 自动装配&#xff1a;根据指定的策略&#xff0c;在IOC容器中匹配某一个bean&#xff0c;自动为指定的bean中所依赖的类类型或者接口类型赋值 之前我们学过的依赖注入&#xff0c;我们在为不同属性赋值时&#xff0c;例如类类型的属性…...

完全二叉树的4种遍历方式

一张二叉树的图 1&#xff0c;二叉树的特点 每个点p的左儿子是p*2,右儿子是p*21&#xff0c;可以分别表示为p<<1与p<<1|1节点的序号是从左到右&#xff0c;从上到下增加的每个点至多2个儿子&#xff08;屁话&#xff08;bushi&#xff09;&#xff09; 2&#xff…...

【vue2】使用elementUI进行表单验证实操(附源码)

&#x1f973;博 主&#xff1a;初映CY的前说(前端领域) &#x1f31e;个人信条&#xff1a;想要变成得到&#xff0c;中间还有做到&#xff01; &#x1f918;本文核心&#xff1a;vue使用elementUI进行表单验证实操&#xff08;附源码&#xff09; 【前言】我们在构建一…...

JUC之阻塞队列解读(BlockingQueue)

目录 BlockingQueue 简介 BlockingQueue 核心方法 1.放入数据 2.获取数据 入门代码案例 常见的 BlockingQueue ArrayBlockingQueue(常用) LinkedBlockingQueue(常用) PriorityBlockingQueue SynchronousQueue LinkedTransferQueue LinkedBlockingDeque 小结 Bloc…...

LCHub:ChatGPT4和低代码来临,程序员面临下岗?

一个网友吐槽道: “ 建站出来了,你们说程序员会失业。 低代码出来了,你们说程序员会失业。 Copilot出来了,你们说程序员会失业。 Chatgpt出来了,你们说程序员会失业 虽然这只是网友的吐槽,但却引起了小编的好奇。为何程序员那么容易被新技术取代?今天小编打算跟大家…...

【Node.js】Express框架的基本使用

✍️ 作者简介: 前端新手学习中。 &#x1f482; 作者主页: 作者主页查看更多前端教学 &#x1f393; 专栏分享&#xff1a;css重难点教学 Node.js教学 从头开始学习 目录 初识Express Express简介 什么是Express 进一步理解 Express Express能做什么 Express的基本使用 …...

使用docker 和 kubnernetes 部署单节点/多节点 kafka 环境

参考资料 https://kafka.apachecn.org/documentation.html#configuration kafka的broker有三个核心配置 broker.idlog.dirszookeeper.connect docker启动单节点kafka环境 启动zookeeper 可配置的环境变量&#xff0c;https://gallery.ecr.aws/bitnami/zookeeper $ docker …...

Linux使用:环境变量指南和CPU和GPU利用情况查看

Linux使用&#xff1a;环境变量指南和CPU和GPU利用情况查看Linux环境变量初始化与对应文件的生效顺序Linux的变量种类设置环境变量直接运行export命令定义变量修改系统环境变量修改用户环境变量修改环境变量配置文件环境配置文件的区别profile、 bashrc、.bash_profile、 .bash…...

深入浅出 SSL/CA 证书及其相关证书文件(pem、crt、cer、key、csr)

互联网是虚拟的&#xff0c;通过互联网我们无法正确获取对方真实身份。数字证书是网络世界中的身份证&#xff0c;数字证书为实现双方安全通信提供了电子认证。数字证书中含有密钥对所有者的识别信息&#xff0c;通过验证识别信息的真伪实现对证书持有者身份的认证。数字证书可…...

网络六边形受到攻击

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 抽象 现代智能交通系统 &#xff08;ITS&#xff09; 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 &#xff08;…...

vscode里如何用git

打开vs终端执行如下&#xff1a; 1 初始化 Git 仓库&#xff08;如果尚未初始化&#xff09; git init 2 添加文件到 Git 仓库 git add . 3 使用 git commit 命令来提交你的更改。确保在提交时加上一个有用的消息。 git commit -m "备注信息" 4 …...

手游刚开服就被攻击怎么办?如何防御DDoS?

开服初期是手游最脆弱的阶段&#xff0c;极易成为DDoS攻击的目标。一旦遭遇攻击&#xff0c;可能导致服务器瘫痪、玩家流失&#xff0c;甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案&#xff0c;帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...

地震勘探——干扰波识别、井中地震时距曲线特点

目录 干扰波识别反射波地震勘探的干扰波 井中地震时距曲线特点 干扰波识别 有效波&#xff1a;可以用来解决所提出的地质任务的波&#xff1b;干扰波&#xff1a;所有妨碍辨认、追踪有效波的其他波。 地震勘探中&#xff0c;有效波和干扰波是相对的。例如&#xff0c;在反射波…...

基于当前项目通过npm包形式暴露公共组件

1.package.sjon文件配置 其中xh-flowable就是暴露出去的npm包名 2.创建tpyes文件夹&#xff0c;并新增内容 3.创建package文件夹...

质量体系的重要

质量体系是为确保产品、服务或过程质量满足规定要求&#xff0c;由相互关联的要素构成的有机整体。其核心内容可归纳为以下五个方面&#xff1a; &#x1f3db;️ 一、组织架构与职责 质量体系明确组织内各部门、岗位的职责与权限&#xff0c;形成层级清晰的管理网络&#xf…...

【项目实战】通过多模态+LangGraph实现PPT生成助手

PPT自动生成系统 基于LangGraph的PPT自动生成系统&#xff0c;可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析&#xff1a;自动解析Markdown文档结构PPT模板分析&#xff1a;分析PPT模板的布局和风格智能布局决策&#xff1a;匹配内容与合适的PPT布局自动…...

【服务器压力测试】本地PC电脑作为服务器运行时出现卡顿和资源紧张(Windows/Linux)

要让本地PC电脑作为服务器运行时出现卡顿和资源紧张的情况&#xff0c;可以通过以下几种方式模拟或触发&#xff1a; 1. 增加CPU负载 运行大量计算密集型任务&#xff0c;例如&#xff1a; 使用多线程循环执行复杂计算&#xff08;如数学运算、加密解密等&#xff09;。运行图…...

Ascend NPU上适配Step-Audio模型

1 概述 1.1 简述 Step-Audio 是业界首个集语音理解与生成控制一体化的产品级开源实时语音对话系统&#xff0c;支持多语言对话&#xff08;如 中文&#xff0c;英文&#xff0c;日语&#xff09;&#xff0c;语音情感&#xff08;如 开心&#xff0c;悲伤&#xff09;&#x…...

华硕a豆14 Air香氛版,美学与科技的馨香融合

在快节奏的现代生活中&#xff0c;我们渴望一个能激发创想、愉悦感官的工作与生活伙伴&#xff0c;它不仅是冰冷的科技工具&#xff0c;更能触动我们内心深处的细腻情感。正是在这样的期许下&#xff0c;华硕a豆14 Air香氛版翩然而至&#xff0c;它以一种前所未有的方式&#x…...