初识优先级队列与堆

1.优先级队列

        由前文队列queue可知,队列是一种先进先出(FIFO)的数据结构,但有些情况下,操作的数据可能带有优先级,一般出队列时,可能需要优先级高的元素先出队列,在此情况下,使用队列queue显然不合适,

        同时在我们生活场景中也有类似于这中优先事情处理的情况,在医院看病的时候大家都在排队,突然有一个头上插了一把刀的老爷爷也来排队,这时候虽然大家都很急着去看病,但是出于老爷爷情况紧急,所以大家都会默认让老爷爷优先去见医生,而不是让老爷爷按照一般的规则去老老实实的排队;

        考虑到随时会遇到上述这种情况,数据结构应该提供两个最基本的操作,一个是返回最高优先级对象一个是添加新的对象。这种数据结构就是优先级队列(Priority Queue)

2.初识堆

        JDK1.8中的PriorityQueue底层使用了这种数据结构,而堆实际就是在完全二叉树的基础上进行了一些调整。

2.1 堆的概念

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

堆的性质

  • 堆中某个节点的值总是不大于或不小于其父节点的值;

  • 堆总是一棵完全二叉树。

2.1.1 小根堆 

                       

        如上图小根堆的逻辑结构所示,在一颗大的完全二叉树中的小二叉树中,该小二叉树的根节点远远小于该树左右子节点,且这时候不考虑左右子节点的数值哪个大;

2.1.2 大根堆

                

         如上图大根堆的逻辑结构所示,给一颗大的完全二叉树中的小二叉树中,该小二叉树的根节点远远大于于该树左右子节点,且这时候不考虑左右子节点的数值哪个大;

2.2 堆的存储方式

        从堆的概念可知,堆是一棵完全二叉树,因此可以层序的规则采用顺序的方式来高效存储,如下图所示堆的存储图解;(堆是将一个一维数组中的数据按照层序遍历的规则将这些数据存储到一个完全二叉树里的完全二叉树)

        注意:由该图中的逻辑结构和存储结构可知,对于非完全二叉树,则不适合使用顺序方式进行存储,因为为了通过一维存储结构能够还原二叉树,一维存储空间中必须要存储空节点,就会导致该存储空间利用率比较低。

         将元素存储到数组中后,可以根据本主初识二叉树章节的性质5对树进行还原。

        假设i为节点在数组中的下标,则有:

        如果i为0,则i表示的节点为根节点,否则i节点的双亲节点为 (i - 1)/2

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

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

2.3 堆的创建(小根堆)

2.3.1 堆向下调整

        思考:

        对于集合{ 27,15,19,18,28,34,65,49,25,37 }中的数据,如何将其创建成堆,首先按照堆的规则将以上数剧放入到完全二叉树里面,如下图所示:

                                 

        仔细观察上图后发现:当前根节点27的左右子树已经完全满足堆的性质,因此只需将根节点向下调整好即可。

        调整过程如下:

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

2. 如果parent的左孩子存在,即:child < size, 进行以下操作,直到parent的左孩子不存在         2.1parent右孩子是否存在,存在找到左右孩子中最小的孩子,让child进行标

     2.2将parent与较小的孩子child比较,如果:

                2.2.1 parent小于较小的孩子child,调整结束

                2.2.2 否则:交换parent与较小的孩子child,交换完成之后,parent中大的元素向下移动,可能导致子树不满足对的性质,因此需要继续向下调整,即parent = child;child = parent*2+1(左子树); 然后继续2

        详细调整图解如下图所示:

2.3.2 小根堆代码实现

package com.bit.demo1;public class MySmallHeap {public void shiftDown(int[] array, int parent) {//child和parent时数组存储的节点的索引// child先标记parent的左孩子,因为parent可能右左没有右int child = 2*parent + 1;int size = array.length;//所有节点的数目while(child < size ) {//主要保证当前访问到的child都在所有节点的范围内,是有效的//child+1是二叉树的最后一个节点// 如果右孩子存在,找到左右孩子中较小的孩子,用child进行标记if(child + 1 < size) {if(array[child + 1] < array[child]) {child = child + 1;}}// 如果最小的孩子比其父亲还小,说明该结构没有满足堆的特性,进行交换if(array[child] < array[parent]) {int tmp = array[parent];array[parent] = array[child];array[child] = tmp;} else {//满足就退出循环break;}// parent中大的元素往下移动,可能会造成子树不满足堆的性质,因此需要继续向下调整parent = child;child = 2*parent + 1;}}}

2.3.3 小根堆代码测试

 public static void main(String[] args) {MySmallHeap mySmallHeap = new MySmallHeap();int[] array = {27,15,19,18,28,34,65,49,25,37};System.out.println("调整前:");for(int i = 0; i < array.length ; i++) {System.out.print(array[i] + " ");}for(int parent = (array.length-2)/2 ; parent >= 0; parent --) {mySmallHeap.shiftDown(array, parent);}System.out.println();System.out.println("调整后:");for(int i = 0; i < array.length ; i++) {System.out.print(array[i] + " ");}System.out.println();}

        关于最初的parent索引 ?

        测试结果如下图所示:

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

        最坏的情况即图示的情况,从根一路比较到叶子,比较的次数为完全二叉树的高度,即时间复杂度为

2.4 建堆的时间复杂度

2.4.1 建堆的步骤图解

        对于普通的序列{ 1,5,3,8,7,6 },我们需要建立大堆,即根节点的左右子树不满足堆的特性,又该如何调整呢,其中里面的细节如下:

        大概思路:

        找倒数第一个非叶子节点(最右侧的最小子树根节点),从该节点位置开始往前进行按照根堆的规则运作,一直运作到根节点,这时候才确定根节点的数值;

        因为为了根节点导致下面的不同子树的结构都发生了变化,所以接下来确定索引为1的根节点的数字,这样就需要重复上述的步骤;

        就这样重复上面两个步骤,知道确定最右侧最小子树的根节点(索引为(arr.length-2)/2)的数值,我们的整体过程才算结束;

2.4.2 时间复杂度求解图解

        至此可得:建堆的时间复杂度为O(N)

ps:本次的内容就到这里了,如果喜欢的话就请一键三连哦!!!

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

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

相关文章

git常用命令指南

目录 一、基本命令 1、创建分支 2、切换分支 3、合并分支 4、初始化空git仓库 二、文件操作 1、创建文件 2、添加多个文件 3、查看项目的当前状态 4、修改文件 5、删除文件 6、提交项目 三、实际操作 1、创建目录 2、进入新目录 3、初始化空git仓库 4、创建文…

C++STL的string模拟实现

文章目录 前言string的成员变量成员函数构造函数拷贝构造赋值重载 模拟实现string各种接口print迭代器普通迭代器const迭代器 string比较大小push_backinsert 和 eraseinserterase reserve和resizereserveresize swapfindcout和cincoutcin 前言 今天要讲string的底层实现&…

总线(什么是南北桥?您都用过哪些总线?)

什么是总线&#xff1f; 计算机系统中的总线&#xff08;Bus&#xff09;是指计算机设备和设备之间传输信息的公共数据通道&#xff0c;是连接计算机硬件系统内多种设备的通信线路&#xff0c;它的一个重要特征是由总线上的所有设备共享&#xff0c;因此可以将计算机系统内的多…

python基于轻量级GhostNet模型开发构建23种常见中草药图像识别系统

轻量级识别模型在我们前面的博文中已经有过很多实践了&#xff0c;感兴趣的话可以自行移步阅读&#xff1a; 《移动端轻量级模型开发谁更胜一筹&#xff0c;efficientnet、mobilenetv2、mobilenetv3、ghostnet、mnasnet、shufflenetv2驾驶危险行为识别模型对比开发测试》 《基…

Vue 核心 数据监听 computed | watch

Vue 核心 数据监听 computed | watch 一、今日学习目标 1.指令补充 指令修饰符v-bind对样式增强的操作v-model应用于其他表单元素 2.computed计算属性 基础语法计算属性vs方法计算属性的完整写法成绩案例 3.watch侦听器 基础写法完整写法 4.综合案例 &#xff08;演示&…

selenium 解决 id定位、class定位中,属性值带空格的解决办法

一、前置说明 selenium遇到下面这种元素&#xff1a; <th id"demo id" class"value1 value2 value3 ">1、虽然id一般不会有空格&#xff0c;但是前端错误的这种写法(如下图)&#xff0c;会造成使用id定位不到元素&#xff0c;如&#xff1a; find…

IOday6作业

1>使用有名管道&#xff0c;完成两个进程的相互通信 //create.c #include<myhead.h>int main(int argc, const char *argv[]) {if((mkfifo("myfifo1",0664)) -1){perror("mkfifo");return -1;}if((mkfifo("myfifo2",0664)) -1){perror…

MYSQL练题笔记-高级查询和连接-这系列最后一题以及下个系列(子查询)的第一题

今天做了下面两题&#xff0c;到第三题的时候想了下但是没有太多的思路&#xff0c;然后看题解的时候实在是觉得自己不会&#xff0c;打算明天看吧。 1.按分类统计薪水相关的表和题目如下 我是想着简化问题&#xff0c;先找出薪水低于30000的员工&#xff0c;然后找这些员工的上…

JAVA 锁

乐观锁 乐观锁是一种乐观思想&#xff0c;即认为读多写少&#xff0c;遇到并发写的可能性低&#xff0c;每次去拿数据的时候都认为别人不会修改&#xff0c;所以不会上锁&#xff0c;但是在更新的时候会判断一下在此期间别人有没有去更新这个数据&#xff0c;采取在写时先读出…

Sam Altman当选“TIME时代周刊”2023年度最佳CEO!还有梅西、Taylor Swift当选...

TIME时代周刊昨日在官网公布了2023年最佳CEO—— Sam Altman当选! 此外&#xff0c;Taylor Swift当选年度最佳人物&#xff0c;梅西当选年度最佳运动员。 Sam Altman的当选可谓是实至名归&#xff01;没有谁能比火爆全球的ChatGPT背后&#xff0c;OpenAI的CEO更“成功”了。 …

ssh安装及问题解决

ssh安装及遇到的问题 ssh分为客户端 openssh-client 和服务器 openssh-server&#xff0c;可以利用以下命令确认是否安装&#xff1a; dpkg -l | grep ssh我用ubantu安装的&#xff0c;所以默认安装了客户端 安装客户端和服务器端的命令分别为&#xff1a; sudo apt-get ins…

金融量化交易:使用Python实现遗传算法

大家好&#xff0c;遗传算法是一种受自然选择过程启发的进化算法&#xff0c;用于寻找优化和搜索问题的近似解决方案。本文将使用Python来实现一个用于优化简单交易策略的遗传算法。 1.遗传算法简介 遗传算法是一类基于自然选择和遗传学原理的优化算法&#xff0c;其特别适用…

MySQL 教程 2.1

MySQL 插入数据 MySQL 表中使用 INSERT INTO 语句来插入数据。 你可以通过 mysql> 命令提示窗口中向数据表中插入数据&#xff0c;或者通过PHP脚本来插入数据。 语法 以下为向MySQL数据表插入数据通用的 INSERT INTO SQL语法&#xff1a; INSERT INTO table_name (colu…

使用Pytorch实现Grad-CAM并绘制热力图

这篇是我对哔哩哔哩up主 霹雳吧啦Wz 的视频的文字版学习笔记 感谢他对知识的分享 看一下这个main cnn.py的文件 那这里我为了方便 就直接从官方的torch vision这个库当中导入一些我们常用的model 比如说我这里的例子是采用的mobile net v3 large这个模型 然后这里我将pretrain设…

微信小程序 纯css画仪表盘

刚看到设计稿的时候第一时间想到的就是用canvas来做这个仪表盘&#xff0c;虽然本人的画布用的不是很好但还可以写一写&#x1f600;。话不多说直接上代码。最后有纯css方法 <!--wxml--> <canvas canvas-id"circle" class"circle" >// js dat…

MySQL 忘记root密码后重置密码操作

在忘记 MySQL 密码的情况下&#xff0c;可以通过 --skip-grant-tables 关闭服务器的认证&#xff0c;然后重置 root 的密码&#xff0c;具体操作步骤如下。 步骤 1)&#xff1a;关闭正在运行的 MySQL 服务。打开 cmd 进入 MySQL 的 bin 目录。 步骤 2)&#xff1a;输入mysqld -…

【Docker】容器数据持久化及容器互联

容器数据持久化及容器互联 一、Docker容器的数据管理1.1、什么是数据卷1.2、数据卷特点1.3、数据卷使用 二、Docker的数据卷容器2.1、什么是数据卷容器2.2、挂载数据卷容器方法 三、Docker数据卷的备份和还原3.1、数据备份方法3.2、数据还原方法 四、Docker容器互联4.1、docker…

xcode ——Instrumets(网络连接调试)使用

环境&#xff1a; instruments 使用只能在真机调试时使用&#xff0c;且真机系统必须ios15 点击debug 按钮——Network——Profile in Instruments 然后就可以看到如下面板 展开运行的项目&#xff0c;点击session下的域名&#xff0c;下方回出现该域名下的网络请求。点击Deve…

管理类联考——数学——真题篇——按题型分类——充分性判断题——秒杀

题型结构 问题求解&#xff1a;通过计算求解&#xff0c;从五个选项中选出一个正确答案。条件充分性判断&#xff1a;问所给的条件&#xff08;1&#xff09;&#xff08;2&#xff09;能否推出题设的结论&#xff0c;共有五个选项&#xff0c;从中选出正确的一个。&#xff0…

LCR 090. 打家劫舍 II(leetcode)动态规划

文章目录 前言一、题目分析二、算法原理1.状态表示2.状态转移方程3.初始化4.填表顺序5.返回值是什么 三、代码实现总结 前言 在本文章中&#xff0c;我们将要详细介绍一下LeetcodeLCR 090. 打家劫舍 II。采用动态规划解决&#xff0c;这是一道经典的多状态dp问题 一、题目分析…