数据结构——优先级队列(PriorityQueue)

1.优先级队列

优先级队列可以看作队列的另一个版本,队列的返回元素是由是由插入顺序决定的,先进先出嘛,但是有时我们可能想要返回优先级较高的元素,比如最大值?这种场景下就由优先级队列登场。
优先级队列底层是由堆实现的,可以说理解了堆就理解了优先级队列。

2.堆的概念

堆可以看作一颗完全二叉树。它将数据以完全二叉树的顺序存储在数组中。堆又分为大根堆小根堆,如果以完全二叉树的方式看待大小根堆。
大根堆就是对于这棵树及该树的每一棵子树都满足左右子树节点值小于根节点值
小根堆就是对于这棵树及该树的每一棵子树都满足左右子树节点值大于根节点值
具体图如下:
在这里插入图片描述

3.实现堆

堆是在一维数组中存储的,所以我们首先需要一个数组去存储我们堆中数据,然后我们还可以通过变量usedSize去表示当前堆中已有元素,具体代码如下:

public class MyHeap {private int [] elem;private int usedSize;public  MyHeap(){//构造方法this.elem=new int[10];this.usedSize=0;}}

为方便演示,我们直接将参数数组中元素“复制”给堆中对应元素。具体代码如下:

public void initHeap(int [] array){for (int i = 0; i <array.length ; i++) {elem[i]=array[i];usedSize++;}}
//运行结果:MyHeap{elem=[10, 26, 98, 75, 15, 64, 85, 32, 12, 100], usedSize=10}

显然此时的堆只能看做为一个数组,其元素按照完全二叉树顺序排列如下图,既不是大根堆也不是小根堆。
在这里插入图片描述

我们要让它变成一个堆就必须对其元素调整,本文演示调整大根堆,那么,如何调整为大根堆?
我们的思路是:从最后一颗子树开始,一颗一颗调整直到根节点,那么对于每一棵树:我们先找到其左右节点的最大值,然后拿着这个最大值跟父节点比较,比父节点大就交换,不如它大就不动。如何找到最后一棵子树:最后一个叶节点对应的父节点,若父节点为p,子节点为i,则有i=2*p+1,p=(i-1)/2。叶子节点下标为usedSize-1,那么其父节点下标就是(usedSize-1-1)/2。所以我们可以写个for循环去遍历调整每棵二叉树。存在一种情况:我在交换父节点和子节点值之后的堆依然不符合大根堆规则,比如上图中75和26换,换完之后26显然比32小,所以还需要再进行交换,我们可以在一次交换过后让parent=child;child=2*parent+1这样继续往下调整,但是我们也不能不限制的往下递归,所以需要一个结束条件,那就是child<usedSize。具体代码如下:

    for(int parent=(usedSize-2)/2;parent>=0;parent--){siftDown(usedSize,parent);}}//针对每一棵树的调整public void siftDown(int usedSize,int parent){int child=2*parent+1;while(child<usedSize){if(child+1<usedSize&&elem[child]<elem[child+1]){child++;}//child指向较大if(elem[child]>elem[parent]){int tmp=elem[child];elem[child]=elem[parent];elem[parent]=tmp;//调整值parent=child;child=2*parent+1;}else{break;}}

offer

作用:此方法用于向堆中添加元素
思路:首先判断堆满没,满了扩容,然后将元素插入到堆中末尾,随后向上调整,将该元素调整到合适位置。
由于usedSize我们可以在类中直接访问到,所以我们push之后的调整只传一个当前末尾节点的下标。然后根据此下标推算出其父节点。若父节点为p,子节点为i,则有i=2*p+1,p=(i-1)/2,如果插入节点大于其父节点就交换,然后child=parent,parent=(child-1)/2,注意此时parent是一级一级往上走的,所以这种调整方式也叫向上调整,同理,刚才那个parent一级一级往下走的就是向下调整。直到根节点遍历完毕或者该插入节点遇到了比它大的父节点就结束循环。具体代码如下:

 public void push(int val){if(isFull()){elem=Arrays.copyOf(elem,2*elem.length);}elem[usedSize]=val;signUp(usedSize);//这里usedSize作为末尾节点下标传进去usedSize++;}private void signUp(int child) {int parent=(child-1)/2;while(parent>-1){if(elem[parent]<elem[child]){//交换int tmp=elem[parent];elem[parent]=elem[child];elem[child]=tmp;//调整child=parent;parent=(child-1)/2;}else {break;}}}private boolean isFull() {return usedSize==elem.length;}

2.2poll

作用:此方法用于拿出堆顶元素
思路:首先将堆顶元素与堆末元素交换,随后将usedSize减一,也就是逻辑上删除了它,等下次push就会覆盖它,但是交换后的堆一定不符合堆的性质,所以还需sfitDown将堆调整为正确的堆。具体代码如下:

 public int poll(){int val=elem[0];int tmp=elem[0];elem[0]=elem[usedSize-1];elem[usedSize-1]=tmp;usedSize--;siftDown(usedSize,0);return val;}

3.优先级队列的基本特性

1.优先级队列中的元素必须是能比较大小的,不然会抛出ClassCastException异常。
2.优先级队列中不能插入null对象,不然会抛出NullPointerException。
3.优先级队列默认实现小根堆。

3.1如何改为大根堆

 PriorityQueue<Integer> queue1=new PriorityQueue<>(Comparator.reverseOrder());

4.top-k问题

给定一组数据,求前K个最大/小的数据。
题目链接:https://leetcode.cn/problems/smallest-k-lcci
思路1:利用优先级队列的特性。看到那句 以任意顺序 我就知道这道题是为优先级队列量身定做的。具体代码实现:

     public int[] smallestK(int[] arr, int k) {PriorityQueue<Integer> queue = new PriorityQueue<>();//非空校验int[] answer = new int[k];//根据题意,注意初始化大小一定要和返回数据数量一致if (arr == null || k <= 0) {int [] a=new int[0];return a;}for(int i=0;i<arr.length;i++){queue.offer(arr[i]);}for(int i=0;i<k;i++){answer[i]=queue.poll();}return answer;}

思路2:用堆排序。
如果求前K个最大值,我们就建小根堆。
如果求前K个最小值,我们就建大根堆。

本题求最小值,我们建大根堆,先把K个数据放到堆中,然后假定它们就是要返回的数据,因为我建的是大根堆,堆顶元素一定是目前堆中最小元素的“最大者”,可以把它理解为“门槛”,然后遍历剩余元素与这个“门槛”比较,如果比“门槛”也就是当前堆顶元素小那我就删除当前堆顶元素同时将这个元素插入进堆,然后继续遍历直到结束,最后返回堆中元素即可。
求最大值思路类似,我建小根堆这样堆顶元素就是当前堆中元素的最小值,如果你比堆顶元素大你就可以入堆,原来的元素被删除。(成王败寇这一块/.)
本题具体代码如下:

public int[] smallestK(int[] arr, int k) {if(arr==null||k<=0){int [] a=new int[0];return a;}PriorityQueue<Integer> queue=new PriorityQueue<>(Comparator.reverseOrder());for(int i=0;i<k;i++){queue.offer(arr[i]);}for(int i=k;i<arr.length;i++){//注意这里i从k开始遍历int peekVal=queue.peek();if(peekVal>arr[i]){queue.poll();queue.offer(arr[i]);}else{continue;}}int [] answer=new int[k];for(int i=0;i<k;i++){answer[i]=queue.poll();}return answer;}

That’s all.

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

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

相关文章

在Windows本地部署Dify详细操作

Dify官网文档&#xff1a;产品简介 - Dify Docs 1.硬件要求 2.部署方式选择 本次我选择Docker Compose 部署&#xff0c;接下来我将根据官方文档指引&#xff0c;在windows电脑上完成dify本地部署 3.DockerCompose本地部署Dify 3.1 安装WSL2 官方安装WSL2的操作说明入口&…

Flutter 与 Android 原生布局组件对照表(完整版)

本对照表用于帮助 Android 开发者快速理解 Flutter 中的布局组件与原生布局的关系。 &#x1f4d8; Flutter ↔ Android 布局组件对照表 Flutter WidgetAndroid View/Layout说明ContainerFrameLayout / View通用容器&#xff0c;可设置背景、边距、对齐等RowLinearLayout (hor…

ps填充图层

在Photoshop&#xff08;PS&#xff09;中&#xff0c;填充图层是一种强大的工具&#xff0c;它允许用户在不破坏原始图像数据的情况下&#xff0c;快速为图像添加颜色、渐变或图案等填充效果。以下从填充图层的类型、创建方法、编辑与修改、应用场景等方面进行详细介绍。 填充…

网页前端开发(基础进阶1--盒子模型)

颜色表示方法3种&#xff1a; 1.关键字&#xff1a; color&#xff1a;green&#xff1b; gray red yellow 2.rgb表示法&#xff1a;红&#xff0c;绿&#xff0c;蓝三原色。rgb&#xff08;r&#xff0c;g&#xff0c;b&#xff09;&#xff0c;r表示红色&#xff0c;g表示绿…

第10讲、Odoo 18框架设计原理全解析

前言 Odoo是一套开源的企业资源规划(ERP)系统&#xff0c;以其模块化、可扩展性和全面的业务应用套件而闻名。Odoo 18作为其最新版本&#xff0c;在架构设计、前端技术和后端实现上都有显著的创新和优化。本文将从前端的OWL组件化、模块化&#xff0c;到后端的ORM封装&#xf…

CSS3 渐变、阴影和遮罩的使用

全文目录&#xff1a; 开篇语**前言****1. CSS3 渐变 (Gradient)****1.1 线性渐变 (linear-gradient)****1.2 径向渐变 (radial-gradient)** **2. CSS3 阴影 (Shadow)****2.1 盒子阴影 (box-shadow)****2.2 文本阴影 (text-shadow)** **3. CSS3 遮罩 (Mask)****3.1 基本遮罩 (m…

[Linux]虚拟地址到物理地址的转化

[Linux]虚拟地址到物理地址的转化 水墨不写bug 文章目录 一、再次认识地址空间二、页表1、页表的结构设计2、页表节省了空间&#xff0c;省在哪里&#xff1f;3、页表的物理实现 一、再次认识地址空间 OS和磁盘交互的内存基本单位是4KB&#xff0c;这4KB通常被称为内存块。OS对…

Kubernetes(K8s)核心架构解析与实用命令大全

在容器化技术席卷全球的今天&#xff0c;Kubernetes&#xff08;简称K8s&#xff0c;以“8”代替“ubernete”八个字母&#xff09;已成为云原生应用部署和管理的核心基础设施。作为Google基于内部Borg系统开源打造的容器编排引擎&#xff0c;K8s不仅解决了大规模容器管理的难题…

基于微信小程序的scratch学习系统

博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业六年&#xff0c;熟悉各种主流语言&#xff0c;精通java、python、php、爬虫、web开发&#xff0c;已经做了六年的毕业设计程序开发&#xff0c;开发过上千套毕业设计程序&#xff0c;没有什么华丽的语言&#xff0…

postgresql 流复制中指定同步的用户

postgresql 流复制中指定同步的用户 在创建postgresql流复制的过程中&#xff0c;可以指定用户名。 主库pg_hba.conf配置 vi $PGDATA/pg_hba.conf host replication repl 192.168.56.12/32 md5 host all all 0.0.0.0/0 md5主库创建同步的用户 # 主库创建 replicator 流复制…

基于springboot的运动员健康管理系统

博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业六年&#xff0c;熟悉各种主流语言&#xff0c;精通java、python、php、爬虫、web开发&#xff0c;已经做了六年的毕业设计程序开发&#xff0c;开发过上千套毕业设计程序&#xff0c;没有什么华丽的语言&#xff0…

工具识别系统Python+深度学习+人工智能+卷积神经网络算法+TensorFlow+图像识别

一、介绍 工具识别系统&#xff0c;使用Python作为主要编程语言&#xff0c;基于TensorFlow搭建卷积神经网络算法&#xff0c;通过收集了8种常见的日常工具图片&#xff08;“汽油罐&#xff08;Gasoline Can&#xff09;”, “锤子&#xff08;Hammer&#xff09;”, “钳子&…

2024 CKA模拟系统制作 | Step-By-Step | 8、题目搭建-创建 Ingress

目录 ​​​​​​免费获取题库配套 CKA_v1.31_模拟系统 一、题目 二、核心考点 Ingress 资源定义 Ingress Controller 依赖 服务暴露验证 网络层次关系 三、搭建模拟环境 1.创建命名空间 2.安装ingress ingress-nginx-controller 3.创建hello.yaml并部署 四、总结 …

关于uv 工具的使用总结(uv,conda,pip什么关系)

最近要开发MCP 项目&#xff0c;uv工具使用是官方推荐的方式&#xff0c;逐要了解这个uv工具。整体理解如下&#xff1a; 一.uv工具的基本情况 UV 是一个由 Rust 编写的现代化 Python 包管理工具&#xff0c;旨在通过极速性能和一体化功能替代传统工具&#xff08;如 pip、vi…

嵌入式学习笔记 - 新版Keil软件模拟时钟Xtal灰色不可更改的问题

在新版Keil软件中&#xff0c;模拟时钟无法修改XTAL频率&#xff0c;默认只能使用12MHz时钟。‌这是因为Keil MDK从5.36版本开始&#xff0c;参数配置界面不再支持修改系统XTAL频率&#xff0c;XTAL选项变为灰色&#xff0c;无法修改。这会导致在软件仿真时出现时间错误的问题&…

Spring AI Image Model、TTS,RAG

文章目录 Spring AI Alibaba聊天模型图像模型Image Model API接口及相关类实现生成图像 语音模型Text-to-Speech API概述实现文本转语音 实现RAG向量化RAGRAG工作流程概述实现基本 RAG 流程 Spring AI Alibaba Spring AI Alibaba实现了与阿里云通义模型的完整适配&#xff0c;…

Java进阶---JVM

JVM概述 JVM作用&#xff1a; 负责将字节码翻译为机器码&#xff0c;管理运行时内存 JVM整体组成部分&#xff1a; 类加载系统(ClasLoader)&#xff1a;负责将硬盘上的字节码文件加载到内存中 运行时数据区(RuntimeData Area)&#xff1a;负责存储运行时各种数据 执行引擎(Ex…

数据类型检测有哪些方式?

typeof 其中数组 对象 null都会判断为Object,其他正确 typeof 2 // number typeof true //bolean typeof str //string typeof [] //Object typeof function (){} // function typeof {} //object typeof undefined //undefined typeof null // nullinstanceof 判断…

NodeJS全栈开发面试题讲解——P6安全与鉴权

✅ 6.1 如何防止 SQL 注入 / XSS / CSRF&#xff1f; 面试官您好&#xff0c;Web 安全三大经典问题分别从不同层面入手&#xff1a; &#x1f538; SQL 注入&#xff08;Server端&#xff09; 原理&#xff1a;恶意用户将 SQL 注入查询语句拼接&#xff0c;导致数据泄露或破坏…

npm error Cannot find module ‘negotiator‘ 的处理

本想运行npm create vuelatest&#xff0c;但提示&#xff1a; npm error code MODULE_NOT_FOUND npm error Cannot find module negotiator npm error Require stack: npm error - C:\Users\Administrator\AppData\Roaming\nvm\v18.16.1\node_modules\npm\node_modules\tuf-j…