【一起学习数据结构与算法】优先级队列(堆)

这篇具有很好参考价值的文章主要介绍了【一起学习数据结构与算法】优先级队列(堆)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、什么是优先级队列?

如果我们给每个元素都分配一个数字来标记其优先级,不妨设较小的数字具有较高的优先级,这样我们就可以在一个集合中访问优先级最高的元素并对其进行查找和删除操作了。这样,我们就引入了 优先级队列 这种数据结构。 优先级队列(priority queue) 是0个或多个元素的集合,每个元素都有一个优先权,对优先级队列执行的操作有(1)查找(2)插入一个新元素 (3)删除 一般情况下,查找操作用来搜索优先权最大的元素,删除操作用来删除该元素 。对于优先权相同的元素,可按先进先出次序处理或按任意优先权进行。–(来源百度)

  1. 普通队列:FIFO。按照元素的入队顺序出队,先入先出。
  2. 按照优先级的大小动态出队(动态指的是元素个数动态变化,而非固定)。

举个现实里面的例子:
排队看病,如果病情相同的情况下会按照先来先进,如果病情严重,优先会看病。
电脑内存占用的资源是有限的,当资源不够的时候,会优先让优先级高的应用占用资源。

二、堆 (heap,基于二叉树)

2.1 什么是堆?

堆在逻辑上是一颗完全二叉树(不存储空节点值),也可以叫二叉堆(Binary Heap)

堆总是攒足下列性质:

  1. 堆中某个结点的值总是不大于或不小于其父结点的值;
  2. 堆总是一棵完全二叉树。

堆是非线性数据结构,相当于一维数组,有两个直接后继。

2.2 堆的分类

将根结点最大的堆叫做最大堆或大根堆,根结点最小的堆叫做最小堆或小根堆。常见的堆有二叉堆、斐波那契堆等。

2.3 结构与存储

从堆的概念可知,堆是一棵完全二叉树,因此可以层序的规则采用顺序的方式来高效存储
如何定义优先级队列,一起学数据结构与算法系列,学习,数据结构,算法
注意:对于非完全二叉树,则不适合使用顺序方式进行存储,因为为了能够还原二叉树,空间中必须要存储空节点,就会导致空间利用率比较低。

三、堆的操作

3.1 堆创建

创建堆之前我们需要复习一个知识点,那就是父子节点之间的关系,

将元素存储到数组中后,可以根据二叉树的性质对树进行还原。假设i为节点在数组中的下标,则有:
如果i为0,则i表示的节点为根节点,否则i节点的双亲节点为 (i - 1)/2
如果2 * i + 1 小于节点个数,则节点i的左孩子下标为2 * i + 1,否则没有左孩子
如果2 * i + 2 小于节点个数,则节点i的右孩子下标为2 * i + 2,否则没有右孩子

复习了这个知识点之后,我们就可以开始思考怎么去创建堆了?
创建的方式其实有很多种,那么我们这里就采用大根堆方式来创建吧!

还记得大根堆是什么特性吗?
将根结点最大的堆叫做最大堆或大根堆。

根据大根堆的特性,我们创建堆的思路是:向下操作。

  1. 需要一个parent标记需要调整的节点。
  2. 设置一个循环条件,使得每次调整都能进行,直至停止。

我们可以用child来标记parent的左孩子,如果存在,判断右孩子是否存在,如果存在找到左右孩子中最大的,依然用child标记,将parent与大孩子比较,parent小于小孩子,交换。

public class TestHeap {
public int[] elem;
    public int usedSize;

    public TestHeap() {
        this.elem = new int[10];
    }

    /**
     * 向下调整函数的实现
     *
     * @param parent 每棵树的根节点
     * @param len    每棵树的调整的结束位置
     */
    public void shiftDown(int parent, int len) {
        int child = 2 * parent + 1;
        // 1. 至少有1个孩子
        while (child < len) {
            if (child + 1 < len && elem[child] < elem[child + 1]) {
                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;
            }
        }
    }

    public void createBigHeap(int[] array) {
        for (int i = 0; i < array.length; i++) {
            elem[i] = array[i];
            usedSize++;
        }

        // 根据代码 显示的时间复杂度 看起来应该是O(N*logn) 但是实际上时O(N)
        for (int parent = (usedSize - 1 - 1) / 2; parent >= 0; parent--) {
            // 调整
            shiftDown(parent, usedSize);
        }
    }
}

3.2 插入元素

插入元素之前,非常经典的我们会考虑到扩容的问题,我们需要判满之后扩容。
扩容之后我们就可以插入新元素了,为了保证堆的结构性,用该位置元素和父亲元素比较,如果大于父亲元素,则交换父子元素,然后指向父亲的位置。再与该位置的父亲位置元素比较,如果父亲元素大则重复上述操作,否则插入结束。

	public void shiftUp(int child) {
        int parent = (child-1)/2;
        while (child > 0) {
            if (elem[child] > elem[parent]) {
                int tmp = elem[child];
                elem[child] = elem[parent];
                elem[parent] = tmp;
                child = parent;
                parent = (child-1)/2;
            } else {
                break;
            }
        }
    }
	public void offer(int val) {
        if (isFull()) {
            // 扩容
            elem = Arrays.copyOf(elem,2*elem.length);
        }
        elem[usedSize++] = val;
        // 注意这里传入的是usedSize-1
        shiftUp(usedSize-1);
    }
    public boolean isFull() {
        return usedSize == elem.length;
    }

3.3 弹出元素

同样在弹出元素之前,我们需要判断是否为空!
如何弹出元素呢?
先将堆尾元素和堆首元素进行交换,然后将usedSize–;之后对堆首元素向下操作即可。

	public boolean isEmpty() {
        return usedSize == 0;
    }

    public int poll() {
        if (isEmpty()) {
            throw new RuntimeException("优先级队列为空!");
        }
        int tmp = elem[0];
        elem[0] = elem[usedSize-1];
        elem[usedSize-1] = tmp;
        shiftDown(0, usedSize);
        return tmp;
    }

四、用堆模拟实现优先级队列

如何定义优先级队列,一起学数据结构与算法系列,学习,数据结构,算法
这里只模拟实现几个重要功能!

public class TestHeap {
    public int[] elem;
    public int usedSize;

    public TestHeap() {
        this.elem = new int[10];
    }

    /**
     * 向下调整函数的实现
     *
     * @param parent 每棵树的根节点
     * @param len    每棵树的调整的结束位置
     */
    public void shiftDown(int parent, int len) {
        int child = 2 * parent + 1;
        // 1. 至少有1个孩子
        while (child < len) {
            if (child + 1 < len && elem[child] < elem[child + 1]) {
                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;
            }
        }
    }

    public void createBigHeap(int[] array) {
        for (int i = 0; i < array.length; i++) {
            elem[i] = array[i];
            usedSize++;
        }

        // 根据代码 显示的时间复杂度 看起来应该是O(N*logn) 但是实际上时O(N)
        for (int parent = (usedSize - 1 - 1) / 2; parent >= 0; parent--) {
            // 调整
            shiftDown(parent, usedSize);
        }
    }

    public void offer(int val) {
        if (isFull()) {
            // 扩容
            elem = Arrays.copyOf(elem,2*elem.length);
        }
        elem[usedSize++] = val;
        // 注意这里传入的是usedSize-1
        shiftUp(usedSize-1);
    }

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

    public boolean isFull() {
        return usedSize == elem.length;
    }

    public boolean isEmpty() {
        return usedSize == 0;
    }

    public int poll() {
        if (isEmpty()) {
            throw new RuntimeException("优先级队列为空!");
        }
        int tmp = elem[0];
        elem[0] = elem[usedSize-1];
        elem[usedSize-1] = tmp;
        shiftDown(0, usedSize);
        return tmp;
    }

    public int peek() {
        if (isEmpty()) {
            throw new RuntimeException("优先级队列为空!");
        }

        return elem[0];
    }

}

五、堆的一个重要应用-堆排序

堆排序(、Heapsort)是指利用堆这种数据结构所设计的一种排序算法。堆是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。
在堆的数据结构中,堆中的最大值总是位于根节点(在优先队列中使用堆的话堆中的最小值位于根节点)。堆中定义以下几种操作:

  1. 最大堆调整(Max Heapify):将堆的末端子节点作调整,使得子节点永远小于父节点
  2. 创建最大堆(Build Max Heap):将堆中的所有数据重新排序
  3. 堆排序(HeapSort):移除位在第一个数据的根节点,并做最大堆调整的递归运算

基本思想:

  1. 将待排序序列构造成一个大顶堆
  2. 此时,整个序列的最大值就是堆顶的根节点
  3. 将其与末尾元素进行交换,此时末尾就为最大值。
  4. 然后将剩余n-1个元素重新构造成一个堆,这样会得到n个元素的次小值。如此反复执行,便能得到一个有序序列了。

1.将堆顶元素8和末尾元素3进行交换
如何定义优先级队列,一起学数据结构与算法系列,学习,数据结构,算法
2.重新调整结构,使其继续满足堆定义
如何定义优先级队列,一起学数据结构与算法系列,学习,数据结构,算法

3.后续过程,继续进行调整,交换,如此反复进行,最终使得整个序列有序

	/**
     * 向下调整函数的实现
     *
     * @param parent 每棵树的根节点
     * @param len    每棵树的调整的结束位置
     */
    public void shiftDown(int parent, int len) {
        int child = 2 * parent + 1;
        // 1. 至少有1个孩子
        while (child < len) {
            if (child + 1 < len && elem[child] < elem[child + 1]) {
                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;
            }
        }
    }
	public void heapSort() {
        int end = this.usedSize-1;
        while (end > 0) {
            int tmp = elem[0];
            elem[0] = elem[end];
            elem[end] = tmp;
            shiftDown(0, end);
            end--;
        }
    }

六、经典的TOPK问题

设计一个算法,找出数组中最小的个数。以任意顺序返回这K个数。

  1. sort排序:取前K个元素
  2. 二叉搜索树:按照中序遍历回收K个数据
  3. 优先级队列:peek,poll

如何定义优先级队列,一起学数据结构与算法系列,学习,数据结构,算法

6.1 排序

对原数组从小到大排序后取出前 kk 个数即可。

class Solution {
    public int[] smallestK(int[] arr, int k) {
        int[] vec = new int[k];
        Arrays.sort(arr);
        for (int i = 0; i < k; ++i) {
            vec[i] = arr[i];
        }
        return vec;
    }
}

  1. 时间复杂度:O(n*log n),其中 n 是数组 arr 的长度。算法的时间复杂度即排序的时间复杂度。
  2. 空间复杂度:O(logn),排序所需额外的空间复杂度为 O(logn)。

如何定义优先级队列,一起学数据结构与算法系列,学习,数据结构,算法

6.2 堆

我们用一个大根堆实时维护数组的前 kk 小值。首先将前 kk 个数插入大根堆中,随后从第 k+1k+1 个数开始遍历,如果当前遍历到的数比大根堆的堆顶的数要小,就把堆顶的数弹出,再插入当前遍历到的数。最后将大根堆里的数存入数组返回即可。

class Solution {
    public int[] smallestK(int[] arr, int k) {
        // 参数检测
        if(null == arr || k <= 0)
            return new int[0];
        PriorityQueue<Integer> q = new PriorityQueue<>(arr.length);
        // 将数组中的元素依次放到堆中
        for(int i = 0; i < arr.length; ++i){
            q.offer(arr[i]);
        }
        // 将优先级队列的前k个元素放到数组中
        int[] ret = new int[k];
        for(int i = 0; i < k; ++i){
            ret[i] = q.poll();
        }
        return ret;
    }
}

  1. 时间复杂度:O(n* logk),其中 nn 是数组 arr 的长度。由于大根堆实时维护前 k 小值,所以插入删除都是 O(logk) 的时间复杂度,最坏情况下数组里 n 个数都会插入,所以一共需要 O(n*logk) 的时间复杂度。
  2. 空间复杂度:O(k),因为大根堆里最多 kk 个数。

如何定义优先级队列,一起学数据结构与算法系列,学习,数据结构,算法文章来源地址https://www.toymoban.com/news/detail-804001.html

到了这里,关于【一起学习数据结构与算法】优先级队列(堆)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【数据结构】优先级队列——堆

    🧧🧧🧧🧧🧧个人主页🎈🎈🎈🎈🎈 🧧🧧🧧🧧🧧数据结构专栏🎈🎈🎈🎈🎈 🧧🧧🧧🧧🧧【数据结构】非线性结构——二叉树🎈🎈🎈🎈🎈 前面介绍过队列,队列是一种先进先出(FIFO)的数据结构,但有些情况下,操作的数据可能带有优先级,一般出队列时,可能

    2024年04月14日
    浏览(48)
  • 数据结构-优先级队列(堆)

    文章目录 目录 文章目录 前言 一 . 堆 二 . 堆的创建(以大根堆为例) 堆的向下调整(重难点)  堆的创建  堆的删除 向上调整 堆的插入 三 . 优先级队列 总结 大家好,今天给大家讲解一下堆这个数据结构和它的实现 - 优先级队列 堆(Heap)是一种基于完全二叉树的数据结构,具有

    2024年02月08日
    浏览(34)
  • 数据结构 之 优先级队列(堆) (PriorityQueue)

    🎉欢迎大家观看AUGENSTERN_dc的文章(o゜▽゜)o☆✨✨ 🎉感谢各位读者在百忙之中抽出时间来垂阅我的文章,我会尽我所能向的大家分享我的知识和经验📖 🎉希望我们在一篇篇的文章中能够共同进步!!! 🌈个人主页:AUGENSTERN_dc 🔥个人专栏:C语言 | Java | 数据结构 ⭐个人

    2024年03月20日
    浏览(34)
  • 数据结构之优先级队列【堆】(Heap)

    目录 1. 优先级队列(Priority Queue) 2.堆的概念 3.堆的存储方式 4.堆的创建 5.用堆模拟实现优先级队列  6.PriorityQueue常用接口介绍 6.1 PriorityQueue的特点 6.2 PriorityQueue几种常见的构造方式 7.top-k问题 8.堆排序 本篇主要内容总结 (1)优先级队列底层是堆来实现的 (2)堆的本质是

    2024年02月01日
    浏览(39)
  • 数据结构 - 6(优先级队列(堆)13000字详解)

    堆分为两种:大堆和小堆。它们之间的区别在于元素在堆中的排列顺序和访问方式。 大堆(Max Heap): 在大堆中,父节点的值比它的子节点的值要大。也就是说,堆的根节点是堆中最大的元素。大堆被用于实现优先级队列,其中根节点的元素始终是队列中最大的元素。 大堆

    2024年02月08日
    浏览(26)
  • 【数据结构】 优先级队列(堆)与堆的建立

    前面介绍过队列, 队列是一种先进先出(FIFO)的数据结构 ,但有些情况下,操作的数据可能带有优先级,一般出队列时,可能需要优先级高的元素先出队列,该中场景下,使用队列显然不合适。 比如:在手机上玩游戏的时候,如果有来电,那么系统应该优先处理打进来的电话

    2024年02月10日
    浏览(28)
  • Java 数据结构篇-用数组、堆实现优先级队列

    🔥博客主页: 【 小扳_-CSDN博客】 ❤感谢大家点赞👍收藏⭐评论✍    文章目录         1.0 优先级队列说明         2.0 用数组实现优先级队列         3.0 无序数组实现优先级队列         3.1 无序数组实现优先级队列 - 入队列 offer(E value)         3.2 无序数组实现优先

    2024年02月04日
    浏览(29)
  • 【数据结构初阶】——第八节.优先级队列(小根堆的模拟实现)

     作者简介:大家好,我是未央; 博客首页: 未央.303 系列专栏:Java初阶数据结构 每日一句:人的一生,可以有所作为的时机只有一次,那就是现在!!! 目录 文章目录 前言 引言 一、堆的概念 二、堆的性质  三、堆的操作 3.1 向下调整算法 3.2 小根堆的创建 3.3 向上调整

    2024年02月07日
    浏览(36)
  • 经典TopK问题、优先级队列 与 堆的纠葛一文为你解惑——数据结构

    前言: 本篇文章以 TopK 问题为引,具体阐述了 PriorityQueue 实现的基本逻辑——堆 数据结构,以及PriorityQueue 的常用方法。如有问题欢迎看官朋友指正,如果觉得文章还不错的话,求点赞、收藏、评论 三连。 重点: 堆的基本实现逻辑 PriorityQueue 运用和源码分析 TopK 问题的解法

    2023年04月22日
    浏览(32)
  • 数据结构学习笔记——图的遍历算法(深度优先搜索和广度优先搜索)

    图的遍历指从图中某一顶点出发(任意一个顶点都可以作为访问的起始顶点),按照某种遍历方法,对图中所有的顶点访问一次且只访问一次。图与树不一样,其中一个顶点可能与多个顶点相连,所以需记录已访问过的顶点,当访问一个顶点后,考虑如何选取下一个要访问的

    2024年02月05日
    浏览(37)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包