【Java--数据结构】优先级队列( PriorityQueue)

一. 优先级队列

1.1  优先级队列的概念

优先级队列是一种特殊的队列,它在入队时会根据元素的优先级进行排序,优先级最高的元素排在队列的前面,出队时会优先出队优先级最高的元素。

1.2 优先级队列的区别

(1)与普通队列的区别

  1. 普通队列是先进先出的,元素按照入队的顺序依次出队。
  2. 优先级队列不考虑入队的先后顺序,只根据元素的优先级来决定出队顺序

(2)与栈的区别

  1. 栈是后进先出的,最后入栈的元素最先出栈。
  2. 优先级队列则根据优先级决定出队顺序,与入队顺序无关。

核心特点元素按优先级排序,每次取出优先级最高(或最低)的元素 

二. 堆

PriorityQueue类底层使用堆这种数据结构,堆适用于完全二叉树

2.1 堆的概念

是一种特殊的完全二叉树数据结构

完全二叉树:除了最后一层外,每一层的节点数都是满的,并且最后一层的节点都靠左排列。

堆的分类:

(1)最大堆:父节点 ≥ 子节点,根节点为全局最大值。

(2)最小堆:父节点 ≤ 子节点,根节点为全局最小值。

2.2 堆的性质

(1)堆序性:

  1. 根节点是堆中的最大元素(大根堆)或最小元素(小根堆)
  • 比如:在大根堆中,根节点(第一层)值大于等于第二层子节点的值,第二层子节点的值又大于等于第三层子节点的值,以此类推。

(2)完全二叉树:

  1. 所有层(除最后一层)必须完全填满

  2. 最后一层的节点必须从左到右连续,不能出现中间空缺

注意:

节点关系下标计算公式
父节点parent = (i-1)/2
左子节点left = 2*i + 1
右子节点right = 2*i + 2

2.3 堆的操作

(1)向下调整

常用于创建堆和删除操作

主要步骤:

循环条件:子节点的下标小于总数

  1. 根据父亲节点找到字节点
  2. 比较得出左右节点的最大节点
  3. 比较父节点和最大节点,如果左右最大节点大于父节点,则交换
  4. 父亲节点等于字节点

代码实现:

    //向下调整public void shiftDown(int i ,int usedSize){int child = (i*2)+1;//左孩子while (child<usedSize) {if (child + 1 < usedSize && elem[child] < elem[child + 1]) {child++;}if (elem[child] > elem[i]) {swap(child,i);i = child;child = (i*2)+1;}else {break;}}}

(2)向上调整

常用于增加操作

 主要步骤:

循环条件:子节点的下标大于0

  1. 找出子节点的父亲节点
  2. 与父亲节点进行大小比较,大于则交换
  3. 让子节点等于父亲

 代码实现:

    //向上调整public void shiftUp(int child){int parent = (child-1)/2;while (child>0) {if (elem[parent] < elem[child]) {swap(parent, child);child = parent;parent = (child - 1) / 2;} else {break;}}}

(3)堆的创建(最大堆)

从最后一个非叶子节点开始,自底向上逐个向下调整

主要步骤:

  1. 得到最大下标的父亲节点(最后一个非叶子节点)
  2. 向下调整

代码实现: 

 public void createHeap(){//找出最大的根节点for (int i = (usedSize-1-1)/2; i >=0 ; i--) {shiftDown(i,usedSize);}}//向下调整public void shiftDown(int i ,int usedSize){int child = (i*2)+1;//左孩子while (child<usedSize) {if (child + 1 < usedSize && elem[child] < elem[child + 1]) {child++;}if (elem[child] > elem[i]) {swap(child,i);i = child;child = (i*2)+1;}else {break;}}}

(4)堆的插入

主要步骤:

  1. 将插入的元素放入最后一位
  2. 向上调整
  3. 堆的元素个数加一个

代码实现:

    public void offer(int val){if(isFull()){this.elem = Arrays.copyOf(elem,elem.length+1);}else {elem[usedSize] = val;shiftUp(usedSize);usedSize++;}}

注意:必须是在已经有序的基础上插入 

(5)堆的删除

堆的删除是删除栈顶元素

主要步骤:

  1. 将栈顶元素和最后一个元素进行交换
  2. 堆中的元素总数减少一个
  3. 栈顶元素向下调整

 代码实现:

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

(6)堆的排序

最大堆——由小到大

主要步骤:

  1. 交换堆顶与末尾元素
  2. 调整剩余堆
  3. 重复直至有序

代码实现:

    //排序从小 到大public void heapSort(){int end = usedSize-1;while(end>0){swap(0,end);shiftDown(0,end);end--;}}

三. PriorityQueue

3.1 PriorityQueue的特性

 注意:

  1. PriorityQueue底层使用堆数据结构
  2. 默认情况下生成的是最下堆,如果想要生成最大堆可以通过使用比较器
  3. 时间复杂度:插入和删除O(log2n)。
  4. 堆中不能存放 null对象
  5. 堆中存放的元素必须要可以比较
  6. 使用时要导入PriorityQueue所在的包

3.2 PriorityQueue的使用

1. 构造方法

(1) 无参构造

Priority( )

        PriorityQueue<Integer> queue = new PriorityQueue<>();

 注意:无参构造方法,没有给定大小,默认容量为11

(2) 有参构造

PriorityQueue(int initialCapacity)

PriorityQueue<Integer> queue1 = new PriorityQueue<>(100);

注意:在调用有参构造器时,传入的参数不能小于1,不然会报错

(3) 利用Collection构建

PriorityQueue(Collection <? extends E>c)

import java.util.PriorityQueue;public class Text_1 {public static void main(String[] args) {PriorityQueue<Integer> queue = new PriorityQueue<>();PriorityQueue<Integer> queue1 = new PriorityQueue<>(100);queue1.offer(12);queue1.offer(3);queue1.offer(9);PriorityQueue<Integer> queue2 =new PriorityQueue<>(queue1);queue2.offer(999);System.out.println(queue2);}
}
//输出:[3, 12, 9, 999]

注意:

在使用PriorityQueue类的时候,不要忘记导入包 

2. 基本操作

(1)插入

插入元素,插入成功返回true,如果插入内容为空,抛出异常

import java.util.PriorityQueue;public class Offer {public static void main(String[] args) {PriorityQueue<Integer> queue = new PriorityQueue<>();queue.offer(14);queue.offer(2);queue.offer(7);System.out.println(queue);}
}

注意: 空间不够时候会自动进行扩容

(2)删除

移除优先级最高的元素并返回,如果优先级队列为空,返回null

import java.util.PriorityQueue;public class Poll {public static void main(String[] args) {PriorityQueue<Integer> queue = new PriorityQueue<>();queue.offer(14);queue.offer(2);queue.offer(7);int x = queue.poll();System.out.println(x);}
}
(3)获取

获取优先级最高的元素,如果优先级队列为空,返回null

import java.util.PriorityQueue;public class Peek {public static void main(String[] args) {PriorityQueue<Integer> queue = new PriorityQueue<>();queue.offer(14);queue.offer(2);queue.offer(7);int x = queue.peek();System.out.println(x);}
}
(4)大小

获取有效元素的个数

import java.util.PriorityQueue;public class Size {public static void main(String[] args) {PriorityQueue<Integer> queue = new PriorityQueue<>();queue.offer(14);queue.offer(2);queue.offer(7);int x = queue.size();System.out.println(x);}
}
(5)清空
import java.util.PriorityQueue;public class IsEmpty {public static void main(String[] args) {PriorityQueue<Integer> queue = new PriorityQueue<>();queue.offer(14);queue.offer(2);queue.offer(7);boolean x = queue.isEmpty();System.out.println(x);queue.clear();boolean x1 = queue.isEmpty();System.out.println(x1);}
}
(6)判空

检测优先级队列是否为空,空返回true

import java.util.PriorityQueue;public class IsEmpty {public static void main(String[] args) {PriorityQueue<Integer> queue = new PriorityQueue<>();queue.offer(14);queue.offer(2);queue.offer(7);boolean x = queue.isEmpty();System.out.println(x);queue.clear();boolean x1 = queue.isEmpty();System.out.println(x1);}
}

3.3 练习

top-k问 题:最大或者最小的前k个数据

求前k最大元素--使用最小堆

求前k最小元素--使用最大堆

(1)求前k个最大元素

方法1:使用最大堆(不推荐,如果数据太多,时间复杂度会很大,很占用内存)

主要步骤:

  1. 将数组内所有元素放入优先级队列中
  2. 每次删除队列中的最大值
  3. 将最大值赋值给新的数组内
    public int[] maxKK(int[] arr, int k) {int[] ret = new int[k];if(arr==null||k<=0){return ret;}//默认最小堆,通过使用比较器,变成默认最大堆PriorityQueue<Integer> queue = new PriorityQueue<>(arr.length);for (int i = 0; i < arr.length; i++) {queue.offer(arr[i]);}//每次弹出的就是最大的for (int i = 0; i < k; i++) {ret[i] = queue.poll();}return ret;}

方法2: 使用最小堆

刷新每一次下限(最小值)

主要步骤:

  1. 先将数组内的前k个元素放进优先级队列里
  2. 从第k个元素往后开始逐个比较,如果k下标的值比队列的最小值大,那就删除最小元素,添加这个元素
  3. 每次都去掉最小的元素,那么就剩下了最大的
    //找出最大的k个数public int[] maxK(int[] array,int k){int[] ret = new int[k];if(k<=0||array == null){return ret;}PriorityQueue<Integer> queue = new PriorityQueue<>(k);for (int i = 0; i < k; i++) {queue.offer(array[i]);}for (int i = k; i < array.length; i++) {int x = queue.peek();if(array[i]>x){queue.poll();queue.offer(array[i]);}}for (int i = 0; i < k; i++) {ret[i] = queue.poll();}return ret;}
(2)求前k个最小元素

使用最大堆

刷新每一次上限(最大值)

主要步骤:

  1. 先将数组内的前k个元素放进优先级队列里
  2. 从第k个元素往后开始逐个比较,如果k下标的值比队列的最大值小,那就删除最大元素,添加这个元素
  3. 每次都去掉最大的元素,那么就剩下了最小的
 public int[] MinK(int[] array,int k){int[] ret = new int[k];if(k<=0||array == null){return ret;}PriorityQueue<Integer> queue = new PriorityQueue<>(new IntCmp());//将数组的前k个元素入队列for (int i = 0; i < k; i++) {queue.offer(array[i]);}for (int i = k; i < array.length; i++) {//队列中的最大值int x = queue.peek();//将最大的弹出去,将这个小的放进来if(array[i] < x){queue.poll();queue.offer(array[i]);}}//将队列中的最大值依次放入数组for (int i = 0; i < k; i++) {ret[i] = queue.poll();}return ret;}

点赞的宝子今晚自动触发「躺赢锦鲤」buff!

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

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

相关文章

【网络编程】HTTP网络编程

13.1 HTTP 简介 HTTP(Hyper Text Transfer Protocol,超文本传输协议)是用于从万维网(WWW:World Wide Web) 服务器(简称Web 服务器)传输超文本到本地浏览器的传送协议&#xff0c;基于TCP/IP 通信协 议来传递数据 (HTML 文件、图片文件、查询结果等)。 13.2 HTTP 的工作原理 …

前端(vue)学习笔记(CLASS 3):生命周期工程化开发入门

1、生命周期 Vue生命周期&#xff1a;一个Vue实例从创建到销毁的整个过程 生命周期四个阶段&#xff1a;创建、挂载、更新、销毁 1、创建阶段&#xff1a;响应式数据 2、挂载阶段&#xff1a;渲染模板 3、更新阶段&#xff1a;数据修改、更新视图&#xff08;执行多次&…

【C++】每日一练(有效的括号)

本篇博客给大家带来的是用C语言来解答有效的括号&#xff01; &#x1f41f;&#x1f41f;文章专栏&#xff1a;每日一练 &#x1f680;&#x1f680;若有问题评论区下讨论&#xff0c;我会及时回答 ❤❤欢迎大家点赞、收藏、分享&#xff01; 今日思想&#xff1a;不服输的少年…

一文讲清楚CUDA与PyTorch、GPU之间的关系

CUDA&#xff08;Compute Unified Device Architecture&#xff09;是由NVIDIA开发的一个并行计算平台和编程模型。它允许软件开发人员和研究人员利用NVIDIA的GPU&#xff08;图形处理单元&#xff09;进行高性能计算。CUDA提供了一系列API和工具&#xff0c;使得开发者能够编写…

Linux:基本指令与内涵理解

1.文件操作指令 1.1 ls ls指令用于查看指定层级文件夹下的文件或文件夹 基本格式&#xff1a;ls (选项) (查看层级&#xff09; 其中选项处不写就默认是显示文件名&#xff0c;查看层级默认是当前层级 选项1&#xff1a; -l 作用&#xff1a;将查找文件的详细信息显示出来 我们…

手机屏幕摔不显示了,如何用其他屏幕临时显示,用来导出资料或者清理手机

首先准备一个拓展坞 然后 插入一个外接的U盘 插入鼠标 插入有数字小键盘区的键盘 然后准备一根高清线&#xff0c;一端链接电脑显示器,一端插入拓展坞 把拓展坞的连接线&#xff0c;插入手机充电口&#xff08;可能会需要转接头&#xff09; 然后确保手机开机 按下键盘…

Unity学习日志番外:简易行为树

Unity简单行为树 参考与代码来自b站-ANVER-大佬教学视频以下都是一种固定模板结构&#xff0c;便于外部以及新项目引用。1.BehaviorTree类2.Node类3.composite4.Sequence5.Selector6.Task7.Blackboard8.实例①兔子行为树②巡逻任务③探测萝卜任务③吃萝卜任务 个人对行为树的理…

【SpringBoot】MD5加盐算法的详解

目录 一、什么是加盐算法 二、如何实现加盐算法 2.1 加盐算法代码实现 2.2 注册页面中进行密码加盐 2.3 登录页面进行加盐的解密 2.4 注册和登录 一、什么是加盐算法 加盐算法是一种用于增强密码安全性的技术。这种技术通过在密码存储过程中添加一个随机生成的盐值&…

【Linux学习笔记】Linux用户和文件权限的深度剖析

【Linux学习笔记】Linux用户和文件权限的深度剖析 &#x1f525;个人主页&#xff1a;大白的编程日记 &#x1f525;专栏&#xff1a;Linux学习笔记 前言 文章目录 【Linux学习笔记】Linux用户和文件权限的深度剖析前言一. Linux权限管理1.1 文件访问者的分类&#xff08;人)…

MinIO问题总结(持续更新)

目录 Q: 之前使用正常&#xff0c;突然使用空间为0B&#xff0c;上传文件也是0B&#xff08;部署在k8s中&#xff09;Q: 无法上传大文件参考yaml Q: 之前使用正常&#xff0c;突然使用空间为0B&#xff0c;上传文件也是0B&#xff08;部署在k8s中&#xff09; A: 1、检查pod状态…

c语言经典基础编程题

c语言经典基础编程题 一、输出输出1.1温度输出1.2排齐数据1.3进制转换 二、选择分支2.1求最大值2.2成绩评定2.3分段函数求值2.4 利润计算2.5判断闰年2.6二次方程根 三、循环结构3.1倒数求和3.2最大数3.3判断素数3.4判断完全数3.5打印菱形&#x1f680;&#x1f680;&#x1f68…

[多线程]基于阻塞队列(Blocking Queue)的生产消费者模型的实现

标题&#xff1a;[多线程]基于阻塞队列(Blocking Queue)的生产消费者模型的实现 水墨不写bug 文章目录 一、生产者消费者模型特点&#xff1a;二、实现2.1详细解释1. 成员变量2. 构造函数3. Isfull 和 Isempty4. Push 函数5. Pop 函数6. 析构函数7. GetSize 函数 三、总结与多线…

在vs中无法用QtDesigner打开ui文件的解决方法

解决方法 右键ui文件&#xff0c;选择打开方式&#xff0c;弹出如下界面。 点击添加&#xff0c;弹出如下界面 点击程序后边的三个点&#xff0c;去电脑查找designer.exe,我的位置为D:\Qt\Qt5.9.9\5.9.9\msvc2015_64\bin\designer.exe。 名称可以自己起一个名字&#xff0c…

[内网渗透] 红日靶场2

环境配置 靶场地址: http://vulnstack.qiyuanxuetang.net/vuln/wiki/ 环境配置可以看这个: https://www.bilibili.com/video/BV1De4y1a7Ps/?spm_id_from333.337.search-card.all.click&vd_sourcecf73ac8de9b7c0322b1bccf77de91c5dNAT模式分配111段, DHCP也要更改 再添加…

第八节:红黑树(初阶)

【本节要点】 红黑树概念红黑树性质红黑树结点定义红黑树结构红黑树插入操作的分析 一、红黑树的概念与性质 1.1 红黑树的概念 红黑树 &#xff0c;是一种 二叉搜索树 &#xff0c;但 在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是 Red和 Black 。 通过对 任何…

基于Spring Boot的网上蛋糕售卖店管理系统的设计与实现(LW+源码+讲解)

专注于大学生项目实战开发,讲解,毕业答疑辅导&#xff0c;欢迎高校老师/同行前辈交流合作✌。 技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、小程序、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容&#xff1a;…

哈尔滨算力服务器托管推荐-青蛙云

哈尔滨年平均气温3.5摄氏度&#xff0c;有发展云计算和算力数据中心的天然优势 &#xff0c;今天为哈尔滨算力服务器托管服务商&#xff1a;青蛙云&#xff0c;黑龙江经营17年的老牌IDC服务商。 先来了解下算力服务器&#xff1a; 算力服务器&#xff0c;尤其是那些用于运行人…

关于Linux contOS 7 的防火墙

centos7 通过firewall-cmd命令添加防火墙白名单 。 查看防护墙状态 firewall-cmd --state 或 systemctl status firewalld active (running)-->表示防火墙已经开启&#xff1b;inactive (dead)-->表示防火墙已经关闭 开关防火墙命令 启动防火墙&#xff1a;systemctl …

【openGauss】物理备份恢复

文章目录 1. gs_backup&#xff08;1&#xff09;备份&#xff08;2&#xff09;恢复&#xff08;3&#xff09;手动恢复的办法 2. gs_basebackup&#xff08;1&#xff09;备份&#xff08;2&#xff09;恢复① 伪造数据目录丢失② 恢复 3. gs_probackup&#xff08;1&#xf…

MySql学习_基础Sql语句

目录 1.数据库相关概念 2.SQL 2.1 SQL通用语法 2.2 SQL分类 2.3 DDL&#xff08;数据库定义语言&#xff09; 2.4 DML&#xff08;数据操作语言&#xff09; 2.5 DQL&#xff08;数据查询语言&#xff09; 2.6 DCL&#xff08;数据控制语言&#xff09; 3. 函数 3.1 字…