优先级队列(堆)详解

优先级队列(堆)详解

目录

  • 堆的概念
  • 堆的存储方式
  • 堆的基本操作
  • 优先级队列模拟实现
  • 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> { 
	@Override
	public int compare(Integer o1, Integer o2) { return o1 - o2; }
 }
//使用比较器创建大根堆
class GreaterIntComp implements Comparator<Integer>{ 
	@Override
	public 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; 
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

动态规划—— 求最长不下降序列LIS【集训笔记】

题目描述 设有由n(1≤n≤200)个整数组成的数列&#xff0c;记为:b(1)、b(2)、……、b(n)&#xff0c;若存在i1<i2<i3<…<ie 且有b(i1)<b(i2)<…<b(ie)则称为长度为e的不下降序列。程序要求&#xff0c;当原数列出之后&#xff0c;求出最长的不下降序列。 …

仰暮计划|“她告诉我,大部分时间她都是一个家庭主妇,负责照料家务和小孩,但她从来没有停止她对知识的追求”

我来到河南省开封市兰考县南北庄村内一个宁静而温馨的小院子&#xff0c;那里居住着一位九十多岁的高龄老人&#xff0c;她就是张奶奶。张奶奶是村里的一位高龄老人&#xff0c;拥有着丰富的人生经历。我对她的故事非常充满好奇&#xff0c;所以特地来到张奶奶的家中&#xff0…

5.Nacos注册中心

5.Nacos注册中心 国内公司一般都推崇阿里巴巴的技术&#xff0c;比如注册中心&#xff0c;SpringCloudAlibaba也推出了一个名为Nacos的注册中心。 5.1.认识和安装Nacos Nacos是阿里巴巴的产品&#xff0c;现在是SpringCloud中的一个组件。相比Eureka功能更加丰富&#xff0c…

【加解密篇】电子数据取证分析之特殊的自加密BitLocker解密

【加解密篇】电子数据取证分析之特殊的自加密BitLocker解密 数据加解密通常是个耗时费力的事情—【蘇小沐】 1、实验环境 Windows 11 专业版&#xff0c;[23H2&#xff08;22631.3007&#xff09;] &#xff08;一&#xff09;自动开启BitLocker之天坑 1、经验之谈 在201…

常用电子器件学习——MOS管

MOS管介绍 MOS&#xff0c;是MOSFET的缩写。MOSFET 金属-氧化物半导体场效应晶体管&#xff0c;简称金氧半场效晶体管&#xff08;Metal-Oxide-Semiconductor Field-Effect Transistor, MOSFET&#xff09;。 一般是金属(metal)—氧化物(oxide)—半导体(semiconductor)场效应晶…

尝试为ssrf漏洞编写黑名单与白名单

以pikachu靶场ssrf&#xff08;curl&#xff09;为例&#xff1a; 你会发现什么也没防御项访问基本的文件内容&#xff0c;端口开放都是可以看到的&#xff0c;没有任何防御措施。 我们去查看一下他的源码有没有过滤什么 没有任何过滤&#xff0c;咱么尝试进行过滤一下&#xf…

P9232 [蓝桥杯 2023 省 A] 更小的数

[蓝桥杯 2023 省 A] 更小的数 终于本弱一次通关了一道研究生组别的题了[普及/提高−] 一道较为简单的双指针题,但一定有更好的解法. 题目描述 小蓝有一个长度均为 n n n 且仅由数字字符 0 ∼ 9 0 \sim 9 0∼9 组成的字符串&#xff0c;下标从 0 0 0 到 n − 1 n-1 n−1&a…

防御第二次作业-防火墙组网实验(2)

目录 实验拓扑图 实验要求 一般组网步骤 to isp区域ping通 dmz区域 trust区域 实验拓扑图 实验要求 1.防火墙向下使用子接口分别对应两个内部区域 2.所有分区设备可以ping通网关 一般组网步骤 1.先配ip、接口、区域、安全策略 2.内网配置回包路由 3.配置dmz区域的服务器映…

工业物联网水泵远程监控系统解决方案

行业描述 水泵作为一种应用极为广泛的通用机械&#xff0c;在工业生产、日常生活中发挥着重要作用&#xff0c;伴随着“十三五”期间重大工程泵类产品的国产化率的提高&#xff0c;当前已经基本满足了国家经济建设发展的需要。 近年来中国水泵企业实现了较大的技术提升&#…

JavaWeb之开发介绍 --黑马笔记

什么是 Web &#xff1f; Web&#xff1a;全球广域网&#xff0c;也称为万维网(www World Wide Web)&#xff0c;能够通过浏览器访问的网站。 Web 网站的工作流程 上图解释&#xff1a; 当你在浏览器中输入网址或点击一个链接时&#xff0c;浏览器会向前端服务器发起请求&…

一文让你了解UI自动化测试

测试都起什么作用 - 是项目的保险&#xff0c;但不是项目的救命草&#xff1b;测试无实际产出&#xff0c;但作用远大于实际产出&#xff1b;测试是从项目维度保证质量&#xff0c;而不是测试阶段。 UI自动化&#xff08;下面简称自动化&#xff09; - 基于UI进行自动功能测试…

贪心算法详解

文章目录 前言一、什么是贪心算法二、贪心算法的特点1.贪心策略的提出2.贪心策略的正确性 三、学习贪心算法的方向总结 前言 在本次文章中我们将会详细介绍贪心算法的相关内容 一、什么是贪心算法 贪心算法&#xff1a;在解决问题时&#xff0c;每一步都选择最优解&#xff0…

K8s(九)持久化存储PV与PVC

PV和PVC PV 和 PVC 之间的关系是一种动态的供需匹配关系。PVC 表示应用程序对持久化存储的需求&#xff0c;而 PV 表示可用的持久化存储资源。Kubernetes 控制平面会根据 PVC 的需求来选择和绑定合适的 PV&#xff0c;将其挂载到应用程序的 Pod 中&#xff0c;从而使应用程序可…

Mock大法:Fake it till u make it!

在单元测试时&#xff0c;我们希望测试环境尽可能单纯、可控。因此我们不希望依赖于用户输入&#xff0c;不希望连接无法独占的数据库或者第三方微服务等。这时候&#xff0c;我们需要通 mock 来模拟出这些外部接口。mock 可能是单元测试中最核心的技术。 无论是 unittest 还是…

【DeepLearning-1】 注意力机制(Attention Mechanism)

1.1注意力机制的基本原理&#xff1a; 计算注意力权重&#xff1a; 注意力权重是通过计算输入数据中各个部分之间的相关性来得到的。这些权重表示在给定上下文下&#xff0c;数据的某个部分相对于其他部分的重要性。 加权求和&#xff1a; 使用这些注意力权重对输入数据进行加权…

Flink(十五)【Flink SQL Connector、savepoint、CateLog、Table API】

前言 今天一天争取搞完最后这一部分&#xff0c;学完赶紧把 Kafka 和 Flume 学完&#xff0c;就要开始做实时数仓了。据说是应届生得把实时数仓搞个 80%~90% 才能差不多找个工作&#xff0c;太牛马了。 1、常用 Connector 读写 之前我们已经用过了一些简单的内置连接器&#x…

用ChatGPT教学、科研!大学与OpenAI合作

亚利桑那州立大学&#xff08;简称“ASU”&#xff09;在官网宣布与OpenAI达成技术合作。从2024年2月份开始&#xff0c;为所有学生提供ChatGPT企业版访问权限&#xff0c;主要用于学习、课程作业和学术研究等。 为了帮助学生更好地学习ChatGPT和大语言模型产品&#xff0c;AS…

禅道的下载使用

文章目录 禅道的下载下载安装包 http://www.zentao.net/安装导南 禅道的使用创建用户产品经理将人员添加进禅道查看权限、产品经理使用禅道添加产品添加产品模块关联用例&#xff08;测试主管&#xff09;执行测试用例转bug 泳道图 禅道的下载 下载安装包 http://www.zentao.n…

电脑无法开机?重装系统教程在这!超详细

#电脑无法开机,怎么重装系统# 前言 本教程适合比较新的Windows电脑硬件。硬件的新旧并没有一个清晰的标准去判定,毕竟有些厂家生产的主板支持UEFI和Legacy两种引导方式,但部分厂家生产的硬件所使用的Bios并不支持Legacy,所以只能用UEFI引导来安装系统。 所以要使用哪种引…

容器原理之Union FS

一、前言 1.1 什么是 UnionFS 联合文件系统&#xff08;UnionFS&#xff09;是一种分层、轻量级并且高性能的文件系统&#xff0c;它支持对文件系统的修改作为一次提交来一层层的叠加&#xff0c;同时可以将不同目录挂载到同一个虚拟文件系统下(unite several directories in…