优先级队列(堆)详解

优先级队列(堆)详解

目录

  • 堆的概念
  • 堆的存储方式
  • 堆的基本操作
  • 优先级队列模拟实现
  • PriorityQueue接口介绍
  • 堆排序
  • Top-k问题

1、堆的概念

如果有一个关键码的集合K = {k0,k1, k2,…,kn-1},把它的所有元素按完全二叉树的顺序存储方式存储 在一个一维数组中,并满足:Ki <= K2i+1 且 Ki<= K2i+2 (Ki >= K2i+1 且 Ki >= K2i+2) i = 0,1,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,否则没有右孩子

3、堆的基本操作

1、向下调整

在讲堆的建立之前,我们要了解一个非常重要的操作:向下调整。
我们以建小根堆为例,给定这样一个完全二叉树:
在这里插入图片描述
我们要把它变成一个小根堆,就要使用到向下调整的方法,对于小根堆我们要保证每一个根结点都小于它的叶子结点(如果有叶子结点)。步骤如下:

  1. 先从最初的根结点开始,记为parent结点,找到其左孩子结点(数组下标2*parent+1)。
  2. 判断有没有右孩子结点,如果存在找到左右孩子中最小的孩子,让child进行标记。
  3. 将parent与较小的孩子child比较,如果:parent小于较小的孩子child,调整结束。
    否则:交换parent与较小的孩子child,交换完成之后,parent中大的元素向下移动,可能导致子树不满足对的性质,因此需要继续向下调整,即parent = child,child = parent*2+1。 然后重复这个过程。直到parent超出数组下标。
    下图是向下调整的过程:在这里插入图片描述
    向下调整的过程也是对于这颗树进行迭代比较而实现的。仔细思考一下,我们发现向下调整的过程其实只影响改变了一条路径,每次遇到孩子结点都有两种情况:1、parent结点不需要和child结点互换,调整直接结束。2、parent结点需要和child结点互换,将child结点作为parent结点继续调整。不需要调整的部分原来就是符合堆的定义,我们只是在这条路径上将最顶部的元素下调至相应位置,这才是parent可以一条路走到黑的根本原因。接下来,我们来看代码(十分巧妙):
  public void shiftDown(int[] array, int parent) {int child = 2*parent+1;//左孩子结点while(child<array.length) {if(child+1< array.length&&array[child]<array[child+1]) {//先判断右孩子存不存在,再找出两个孩子中的较大孩子child+=1;//child指向较大的孩子结点}if(array[parent]>array[child]) {//比最大的孩子还大,向下调整swap(array,parent,child);}else{break;}parent = child;//继续向下调整child = 2*parent+1;}

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

2、堆的创建

对于一个普通序列:根节点的左右子树不满足堆的特性,我们又该如何调整呢?
答案仍然是利用向下调整,只不过我们要从下向上考虑。步骤如下:

  1. 从后向前,找到第一个不为叶子结点的结点,记为root。
  2. 从这个根结点向前遍历数组(root–),每遇到一个结点就向下调整一次,直到走完数组(root<0)。
    直接上代码:
  public static void createHeap(int[] array) {int root = (array.length-1-1)/2;//根据孩子结点求根结点for(;root>=0;root--) {shiftDown(array,root);}}

3、堆的插入

堆的插入总共需要两个步骤:

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

这时,我们所做的实际上是一种向上调整。仍然以小根堆为例,下面是代码实现:

public static void shiftUp(int[] array,int child) {int parent = (child-1)/2;while(child>0) {if(array[parent]>array[child]) {swap(array,parent,child);}else{break;}child = parent;parent = (child-1)/2;}}

比较向上调整与向下调整:
1、向上调整是通过child结点去找parent结点,而向下调整是通过parent结点去找child结点。
2、建堆的时候使用向下调整,而不使用向上调整。因为向下调整操作的复杂度较低。

4、堆的删除

堆的删除一定删除的是堆顶元素。具体如下:

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

4、优先级队列模拟实现

前面介绍过队列,队列是一种先进先出(FIFO)的数据结构,但有些情况下,操作的数据可能带有优先级,一般出队列时,可能需要优先级高的元素先出队列
在这种情况下,数据结构应该提供两个最基本的操作:一个是返回最高优先级对象,一个是添加新的对象。这种数据结构就是优先级队列(Priority Queue)。
我们接下来使用堆的操作模拟实现优先级队列,直接上代码:

import java.util.Arrays;public class PriorityQueue {public int[] array;public int usedSize;public static final int default_Capacity = 10;//默认容量public PriorityQueue() {this.usedSize = 0;this.array = new int[default_Capacity];}/**** @param array*/public void createHeap(int[] array) {int parent = (array.length-2)/2;for(;parent>=0;parent--) {shiftDown(parent,usedSize);}}public static void swap (int[] array,int parent,int child) {int tmp = array[parent];array[parent] = array[child];array[child] = tmp;}/**** @param parent 是每棵子树的根节点的下标* @param len  是每棵子树调整结束的结束条件* 向下调整的时间复杂度:O(logn)*/private void shiftDown(int parent,int len) {int child = 2*parent+1;while(child<len) {if(child+1<len&&array[child]<array[child+1]) {child+=1;//child指向较大的孩子结点}if(array[parent]>array[child]) {swap(array,parent,child);}else{break;}parent = child;child = 2*parent+1;}}/*** 入队:仍然要保持是大根堆* @param val*/public void push(int val) {if(isFull()) {array = Arrays.copyOf(array,2*array.length);}array[usedSize] = val;usedSize++;shiftUp(usedSize-1);}private void shiftUp(int child) {int parent = (child-1)/2;while(child>0) {if(array[parent]>array[child]) {swap(array,parent,child);}else{break;}child = parent;parent = (child-1)/2;}}public boolean isFull() {return usedSize == array.length;}/*** 出队【删除】:每次删除的都是优先级高的元素* 仍然要保持是大根堆*/public int pollHeap() {int tmp = array[0];swap(array,0,usedSize-1);usedSize--;shiftDown(0,usedSize);return tmp;}public boolean isEmpty() {return usedSize==0;}/*** 获取堆顶元素* @return*/public int peekHeap() {return array[0];}
}

5、PriorityQueue接口介绍

三种构造器:

  • PriorityQueue() 创建一个空的优先级队列,默认容量是11
  • PriorityQueue(int initialCapacity)创建一个初始容量为initialCapacity的优先级队列,注意:
    initialCapacity不能小于1,否则会抛IllegalArgumentException异常
  • PriorityQueue(Collection<?extends E> c) 用一个集合来创建优先级队列

常用方法

  • booleanoffer(E e)插入元素e,插入成功返回true,如果e对象为空,抛出NullPointerException异常,时间复杂度为O(log2N) ,注意:空间不够时候会进行扩容
  • E peek() 获取优先级最高的元素,如果优先级队列为空,返回null
  • E poll() 移除优先级最高的元素并返回,如果优先级队列为空,返回null
  • int size() 获取有效元素的个数
  • void clear() 清空队列
  • boolean isEmpty() 检测优先级队列是否为空,空返回true

6、堆排序

堆排序即利用堆的思想来进行排序,总共分为两个步骤:

  1. 建堆
    升序:建大堆
    降序:建小堆
  2. 利用堆删除思想来进行排序
    建堆和堆删除中都用到了向下调整,因此掌握了向下调整,就可以完成堆排序。
   public void heapSort() {int end = usedSize-1;while (end > 0) {swap(0,end);siftDown(0,end-1);end--;}

7、Top-k问题(堆排序的应用)

TOP-K问题:即求数据集合中前K个最大的元素或者最小的元素,一般情况下数据量都比较大。
基本思路如下:

  1. 用数据集合中前K个元素来建堆
    前k个最大的元素,则建小堆
    前k个最小的元素,则建大堆
  2. 用剩余的N-K个元素依次与堆顶元素来比较,不满足则替换堆顶元素
    将剩余N-K个元素依次与堆顶元素比完之后,堆中剩余的K个元素就是所求的前K个最小或者最大的元素。
//使用比较器创建小根堆
class LessIntComp implements Comparator<Integer> { @Overridepublic int compare(Integer o1, Integer o2) { return o1 - o2; }}
//使用比较器创建大根堆
class GreaterIntComp implements Comparator<Integer>{ @Overridepublic int compare(Integer o1, Integer o2) { return o2 - o1; } 
}
public class TestDemo<E> { //求最小的K个数,通过比较器创建大根堆public static int[] smallestK(int[] array, int k) { if(k <= 0) { return new int[k]; }GreaterIntComp greaterCmp = new GreaterIntComp(); PriorityQueue<Integer> maxHeap = new PriorityQueue<>(greaterCmp); //先将前K个元素,创建大根堆 for(int i = 0; i < k; i++) {maxHeap.offer(array[i]); }//从第K+1个元素开始,每次和堆顶元素比较 for (int i = k; i < array.length; i++) { int top = maxHeap.peek(); if(array[i] < top) {maxHeap.poll();maxHeap.offer(array[i]); } }//取出前K个int[] ret = new int[k]; for (int i = 0; i < k; i++) { int val = maxHeap.poll(); ret[i] = val;}return ret; 
}

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

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

相关文章

SAP 五个报废率设置简介(上)

通常在生产制造过程中都会面临报废率的问题,生产工艺路线的问题,原材料质量的问题,总会有一些产品在生产过程中被做成报废品,通常报废率的设置有时候会遵循行业的标准设置,亦或者根据工厂生产中统计的历史数据分析后根据不同的产品设置不同的报废率,从而在执行物料的采购…

VM下Unbunt虚拟机上网设置

系列文章目录 VM下虚拟机上网设置 VM下虚拟机上网设置 右击VM软件中你需要设置的虚拟机&#xff0c;选择设置 宿主机如果你用的是笔记本外加WIFI连接选择NAT网络模式 进入虚拟机看能否上网 不行的话&#xff0c;进入虚拟机点击&#xff0c;选择最后一栏&#xff0c;编辑连接 点…

华为认证的HCIP考实验考试么?

HCIP在考试的时候不考实验&#xff0c;只考笔试&#xff0c;只是不同方向的HCIP认证考试的考试科目不同&#xff0c;有的考一科&#xff0c;有的考二科&#xff0c;有的考三科&#xff0c;具体看方向来定。HCIA和HCIP只考笔试。HCIE考笔试和实验。 虽然HCIP不考实操&#xff0…

《WebKit 技术内幕》学习之七(1): 渲染基础

《WebKit 技术内幕》之七&#xff08;1&#xff09;&#xff1a; 渲染基础 WebKit的布局计算使用 RenderObject 树并保存计算结果到 RenderObject 树。 RenderObject 树同其他树&#xff08;如 RenderLayer 树等&#xff09;&#xff0c;构成了 WebKit 渲染的为要基础设施。 1…

【数据结构】链表(单链表与双链表实现+原理+源码)

博主介绍&#xff1a;✌全网粉丝喜爱、前后端领域优质创作者、本质互联网精神、坚持优质作品共享、掘金/腾讯云/阿里云等平台优质作者、擅长前后端项目开发和毕业项目实战✌有需要可以联系作者我哦&#xff01; &#x1f345;附上相关C语言版源码讲解&#x1f345; &#x1f44…

python04-变量命名规则

python需要使用标识符来给变量命名。 标识符&#xff0c;我来解释下&#xff0c;就是给程序中变量、类、方法命名的符号&#xff0c;简单理解就是起一个名字&#xff0c;这个名字必须是合法的名字&#xff0c; 对于Python来说&#xff0c;标识符必须是以字母、下划线(_)开头&…

鸿蒙自定义刷新组件使用

前言 DevEco Studio版本&#xff1a;4.0.0.600 1、RefreshLibrary_HarmonyOS.har&#xff0c;用于HarmonyOS "minAPIVersion": 9, "targetAPIVersion": 9, "apiReleaseType": "Release", "compileSdkVersion": "3.…

用户画像系列——在线服务调优实践

前面文章讲到画像的应用的几个方面&#xff0c;其中画像的在线服务应用主要是在推荐场景、策略引擎场景&#xff0c;这两部分场景都是面向线上的c端服务。 推荐场景&#xff1a;根据不同的用户推荐不同的内容&#xff0c;做到个性化推荐&#xff0c;需要读取画像的一些偏好数据…

【方法】如何把Excel“只读方式”变成可直接编辑?

Excel在“只读方式”下&#xff0c;编辑后是无法直接保存原文件的&#xff0c;那如何可以直接编辑原文件呢&#xff1f;下面来一起看看看吧。 如果Excel设置的是无密码的“只读方式”&#xff0c;那在打开Excel后&#xff0c;会出现对话框&#xff0c;提示“是否以只读方式打开…

什么是甘特图?谁还不知道?做管理的来看看!

在现代商业社会&#xff0c;项目管理已成为不可或缺的技能。而甘特图作为一种强大的项目管理工具&#xff0c;正逐渐受到越来越多人的青睐。那么&#xff0c;什么是甘特图&#xff1f;又有什么工具可以绘制甘特图呢&#xff1f;本文将为你一一解答。 一、甘特图的定义 甘特图…

Unity - 简单音频视频

“Test_04” 音频 使用AudioTest脚本控制Audio Source组件&#xff0c;在脚本中声明"music"和"se"之后&#xff0c;在unity中需要将音频资源拖拽到对应位置。 AudioTest public class AudioTest : MonoBehaviour {// 声明音频// AudioClippublic AudioC…

Vulnhub-dc4

靶场下载 https://download.vulnhub.com/dc/DC-4.zip 信息收集 判断目标靶机的存活地址: # nmap -sT --min-rate 10000 -p- 192.168.1.91 -oN port.nmap Starting Nmap 7.94 ( https://nmap.org ) at 2024-01-21 16:36 CST Stats: 0:00:03 elapsed; 0 hosts completed (1 up…

三、MySQL之创建和管理表

一、基础知识 1.1 一条数据存储的过程 存储数据是处理数据的第一步 。只有正确地把数据存储起来,我们才能进行有效的处理和分析。否则,只 能是一团乱麻,无从下手。 在 MySQL 中, 一个完整的数据存储过程总共有 4 步,分别是创建数据库、确认字段、创建数据表、插入数据。 …

在字节5年被优化,太难了。。。

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 先简单说下&#xff0c;涵哥是某不知名 985 的本硕&#xff0c;17 年毕业加入字节&#xff0c;以…

自学网安-DNS

01DNS Domain Name Service域名服务 作用&#xff1a;为客户机提供域名解析服务器 02域名组成 2.1域名组成概述 如"www.sina.com.cn"是一个域名&#xff0c;从严格意义上讲&#xff0c;"sina.com.cn"才被称为域名(全球唯一)&#xff0c;而"www"…

layui 自定义日期选择器今日、昨日 、本周、本月、上个月等

1、layui 日期选择器 laydate日期选择器 <div class"layui-input-inline"><input class"layui-input" id"dateTime" placeholder"日期范围"> </div><script> layui.use([laydate], function () {laydate.ren…

QT upd测试

QT upd测试 本次测试将服务器和客户端写在了一个工程下&#xff0c;代码如下 widget.h #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include<QUdpSocket> #include<QTimer>QT_BEGIN_NAMESPACE namespace Ui { class Widget; } QT_END_NAMESPACE…

编程入门:五个你必须知道的编程常识

常识1&#xff1a;编程不仅仅是写代码 当我们谈论编程时&#xff0c;大多数人首先想到的是写代码。这是正确的&#xff0c;但并不完整。编程不仅仅是写代码&#xff0c;而是解决问题的一种方式。编程是一种工具&#xff0c;我们使用它来创建能够解决特定问题的产品和服务。 比…

qmt和ptrade有什么区别?国内免费量化交易软件精选:让你轻松上手量化交易!

QMT 和 PTrade 是两个不同的平台&#xff0c;具有不同的功能和特点。 QMT&#xff08;Quantitative Market Trading&#xff09;是一种量化交易平台&#xff0c;主要面向专业的量化交易员和机构交易员。它提供了一系列的工具和功能&#xff0c;帮助交易员进行定量分析、模型开…

C#用DateTime.Now静态属性返回日期的星期信息

目录 一、使用的方法 1.Now属性 2.ToString方法 二、示例 使用DateTime结构的Now静态属性&#xff0c;可以方便地获取系统日期信息。调用时间对象的ToString方法&#xff0c;在该方法的参数中添加适当的格式化字符串&#xff0c;将返回日期的星期信息。 一、使用的方法 1…