初识优先级队列与堆

news2024/11/28 3:52:36

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.coloradmin.cn/o/1295891.html

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

相关文章

xilinx的XVC协议

文章目录 概述JTAG工作方式XVC协议 其他Debug Bridge IP 概述 JTAG工作方式 XVC协议 其他 Debug Bridge IP

cookie总结

cookie和session&#xff1a; 一、Cookie和Session二、使用Cookie保存用户上次的访问时间。三、Cookie常用方法总结乱码问题解决&#xff1a; 一、Cookie和Session 会话&#xff1a;用户从打开浏览器到关闭的整个过程就叫1次会话。 比如有的网站登录过一次&#xff0c;下次再进…

python的websocket方法教程

WebSocket是一种网络通信协议&#xff0c;它在单个TCP连接上提供全双工的通信信道。在本篇文章中&#xff0c;我们将探讨如何在Python中使用WebSocket实现实时通信。 websockets是Python中最常用的网络库之一&#xff0c;也是websocket协议的Python实现。它不仅作为基础组件在…

Spring AOP 概念及其使用

目录 AOP概述 什么是AOP&#xff1f; 什么是Spring AOP ? Spring AOP 快速入门 1.引⼊ AOP 依赖 2.编写AOP程序 Spring AOP 核心概念 1.切点 2.连接点 3.通知 4.切面 通知类型 注意事项: PointCut&#xff08;定义切点&#xff09; 切面优先级 Order 切点表达…

IDEA删除最近打开的文件记录

IDEA删除最近打开的文件记录 遇见问题&#xff1a;如何删除IDEA中最近打开的文件记录 解决方法 先关闭IDEA 找到 recentProjects.xml 文件 windows 位置&#xff1a;&#xff08;AppData是隐藏文件夹&#xff09; 1.C:\Users\电脑用户名\AppData\Roaming\JetBrains\IntelliJIde…

Bash脚本调用百度翻译API进行中文到英文的翻译

写一个bash脚本调用百度翻译API进行中文到英文的翻译&#xff0c;首先需要进行相关的申请。看百度给出的文档链接: 百度翻译API文档 需要先注册一个百度账号&#xff0c;然后申请APPID。脚本中会用到appid和key这两个值。按照文档给出的提示可以获得。如下是脚本&#xff1a; #…

零基础如何入门HarmonyOS开发?

HarmonyOS鸿蒙应用开发是当前非常热门的一个领域&#xff0c;许多人都想入门学习这个技术。但是&#xff0c;对于零基础的人来说&#xff0c;如何入门确实是一个问题。下面&#xff0c;我将从以下几个方面来介绍如何零基础入门HarmonyOS鸿蒙应用开发学习。 一、了解HarmonyOS鸿…

markdown记录

文章目录 基础操作使用一级列表、二级列表 博文链接 基础操作 使用一级列表、二级列表 博文链接 CSDN-Markdown语法集锦 CSDN-markdown语法之如何使用LaTeX语法编写数学公式 CSDN Markdown简明教程1-关于Markdown CSDN Markdown简明教程2-基本使用 CSDN Markdown简明教程3-表…

X86汇编语言:从实模式到保护模式(代码+注释)--c6

X86汇编语言&#xff1a;从实模式到保护模式&#xff08;代码注释&#xff09;–c6 标志寄存器FLAGS&#xff1a; 6th&#xff1a;ZF位&#xff08;Zero Flag&#xff09;&#xff1a;零标志&#xff0c;执行算数或者逻辑运算之后&#xff0c;会将该位置位。10th&#xff1a;D…

油猴(Tampermonkey)浏览器插件简单自定义脚本开发

介绍 浏览器插件&#xff0c;包括油猴插件和其他插件&#xff0c;通过它们可以实现浏览器网页的定制化与功能增强。 其他插件一般只有某种具体的功能&#xff0c;且已经写死而不能更改&#xff0c;比如Adblock插件只用于去广告。 油猴插件是一款用于管理用户脚本的插件&…

RocketMQ - SpringBoot整合RocketMQ

SpringBoot整合RocketMQ 1、快速实战 ​ 按照SpringBoot三板斧&#xff0c;快速创建RocketMQ的客户端。创建Maven工程&#xff0c;引入关键依赖&#xff1a; <dependencies><dependency><groupId>org.apache.rocketmq</groupId><artifactId>r…

HarmonyOS应用开发工具DevEco Studio安装与使用

语雀知识库地址&#xff1a;语雀HarmonyOS知识库 飞书知识库地址&#xff1a;飞书HarmonyOS知识库 知识库内容逐步完善中… 工欲善其事必先利其器&#xff0c;要编写HarmonyOS应用就需要用到官方提供的IDE工具来编写相应的代码。 在鸿蒙开发者官网&#xff0c;其提供了官方的开…

Markdown从入门到精通

Markdown从入门到精通 文章目录 Markdown从入门到精通前言一、Markdown是什么二、Markdown优点三、Markdown的基本语法3.1 标题3.2 字体3.3 换行3.4 引用3.5 链接3.6 图片3.7 列表3.8 分割线3.9 删除线3.10 下划线3.11 代码块3.12 表格3.13 脚注3.14 特殊符号 四、Markdown的高…

Ubuntu 22.04源码安装yasm 1.3.0

sudo lsb_release -r看到操作系统的版本是22.04&#xff0c;sudo uname -r可以看到内核版本是5.15.0-86-generic&#xff0c;sudo gcc --version可以看到版本是11.2.0&#xff0c;sudo make --version可以看到版本是GNU Make 4.3。 下载yasm http://yasm.tortall.net/Downlo…

深入了解Java 8日期时间新玩法:DateTimeFormatter与ZoneOffset的使用

推荐语 在这篇文章中&#xff0c;我们将深入探讨Java中的DateTimeFormatter和ZoneOffset类的功能和使用方法。这些类是在Java 8中引入的新的日期时间API的一部分&#xff0c;它们为我们提供了更灵活、更易用的日期和时间处理能力。尽管这些类在Java 8中已经出现&#xff0c;但…

SIT3232E高静电防护,单电源供电,双通道,RS232 收发器

SIT3232E 是一款 3.0V~5.5V 供电、双通道、低功耗、高静电防护 ESD 保护&#xff0c;完全满足 TIA/EIA-232 标准要求的 RS-232 收发器。 SIT3232E 包括两个驱动器和两个接收器&#xff0c;具有增强形 ESD 保护功能&#xff0c;达到 15kV 以上 HBM ESD 、 8kV …

拼多多商品详情数据接口在数据分析行业的作用性

在数据分析行业中&#xff0c;拼多多商品详情数据的作用性主要体现在以下几个方面&#xff1a; 了解市场和用户需求&#xff1a;通过拼多多商品详情数据&#xff0c;企业可以了解到市场上什么产品受欢迎&#xff0c;用户对产品的反馈和评价如何&#xff0c;从而调整自己的销售…

Qt/QML编程学习之心得:工程中的文件(十二)

Qt生成了工程之后,尤其在QtCreator产生对应的project项目之后,就如同VisualStudio一样,会产生相关的工程文件,那么这些工程文件都是做什么的呢?这里介绍一下。比如产生了一个Qt Widget application,当然如果Qt Quick Application工程会有所不同。 一、.pro和.pro.user …

神经内科临床常用的焦虑、抑郁评估量表,医生必备!

根据神经内科医生的量表使用情况&#xff0c;常笑医学整理了神经内科临床上常用的焦虑、抑郁评估量表&#xff0c;为大家分享临床常见的焦虑、抑郁、睡眠等量表评估内容&#xff0c;均支持量表下载和在线使用&#xff0c;建议收藏&#xff01; 1.汉密顿抑郁量表&#xff08;Ham…

初识Linux:权限(1)

目录 提示&#xff1a;以下指令均在Xshell 7 中进行 Linux 的权限 内核&#xff1a; 查看操作系统版本 查看cpu信息 查看内存信息 外部程序&#xff1a; 用户&#xff1a; 普通用户变为超级用户&#xff1a; su 和 su-的区别&#xff1a; root用户变成普通用户&#…