【数据结构初阶】——第八节.优先级队列(小根堆的模拟实现)

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

 作者简介:大家好,我是未央;

博客首页:未央.303

系列专栏:Java初阶数据结构

每日一句:人的一生,可以有所作为的时机只有一次,那就是现在!!!

目录

文章目录

前言

引言

一、堆的概念

二、堆的性质 

三、堆的操作

3.1 向下调整算法

3.2 小根堆的创建

3.3 向上调整算法

3.4 堆的删除(堆顶元素的删除)

四、优先级队列的模拟实现(小根堆)

总结


今天我们将进入到有关堆的有关内容的学习,以及有关优先级队列的相关使用,要对堆的概念,性质,操作有很熟悉的认识和了解,接下来就让我们进入到今天的学习当中吧!!!!!!


引言

我们之前学过队列,那么什么是优先级队列呢?

举个例子

队列是一种先进先出(FIFO)的数据结构但是有些情况下,操作的数据可能带有优先级,一般出队列时,可能需要优先级高的元素先出队列,在这种情况下使用队列就不行了,比如玩游戏的时候突然女朋友一通电话,游戏屏幕瞬间被电话占领,这时候就应该优先处理电话。

在这种情况下,我们的数据结构应该提供两个最基本的操作,一个是返回最高优先级对象,一个是添加新对象,这种数据结构就是优先级队列(PriorityQueue)。

但其实这种对优先级队列的定义是不严谨的,严谨的说法是:

优先级队列是逻辑结构是小根堆存储结构是动态数组(到达上限,容量自动加一)的集合类。
所以说优先级队列不是数据结构中的概念,而是java中的集合类。
但在JDK1.8中的PriorityQueue底层使用了堆的数据结构,而堆实际就是在完全二叉树的基础之上进行了一些元素的调整,所以说优先级队列是一种数据结构也未尝不可。

 既然优先级队列的底层用到了堆这种数据结构,那么什么是堆呢?

小根堆,Java初阶数据结构,数据结构,堆,java

一、堆的概念

前提知识:二叉树的顺序存储

使用数组存储二叉树的方式,就是将二叉树按照层序遍历放入数组;

一般只适合完全二叉树,因为非完全二叉树会有空间的浪费;

小根堆,Java初阶数据结构,数据结构,堆,java

 而堆其实就是一棵完全二叉树,所以就可以用数组来储存堆这一数据结构;

总结:堆就是一颗顺序存储的完全二叉树,底层是一个数组;

  1. 堆逻辑上是一颗完全二叉树

  2. 堆物理上是保存在数组中

小根堆,Java初阶数据结构,数据结构,堆,java

二、堆的性质 

既然是完全二叉树,那么之前我们得出来的那些完全二叉树的性质也同样适用于堆;

下面是一个完全二叉树,也同样是一个堆小根堆,Java初阶数据结构,数据结构,堆,java

也就是说已知父亲结点的下标就能求出孩子结点的下标,同理如果知道了孩子结点的下标也能求出他所对应的父亲结点的下标;

堆的分类:大根堆和小根堆;

  1. 小根堆:每一个父结点的值均小于等于其对应的子结点的值,而根结点的值就是最小的。
  2. 大根堆:每一个父结点的值均大于等于其对应的子结点的值,而根结点的值就是最大的。

图示说明:

小根堆,Java初阶数据结构,数据结构,堆,java

如此一来我们就又得到了堆的另外两条性质:

  1. 堆中某个节点的值总是不大于或不小于其父节点的值。
  2. 堆总是一棵完全二叉树。
  3. 但完全二叉树不一定能称作堆(可能无序,不满足堆的性质)

小根堆,Java初阶数据结构,数据结构,堆,java

三、堆的操作

3.1 向下调整算法

图示说明:

小根堆,Java初阶数据结构,数据结构,堆,java

 此时我们看到,这个二叉树整体上不符合堆的性质,但是其根部的左子树和右子树均满足堆的性质。 

注意上面说的也是向下调整的前提即——必须得确保根结点的左右子树均为小堆才可;

接下来,就要进行向下调整,确保其最终是个堆。只需三步。

  1. 找出左右孩子中最小的那个
  2. 跟父亲比较,如果比父亲小,就交换
  3. 更新父亲结点(继续向下调整,因为经过上一次调整后下面的子树可能就不满足小根堆的性质了)

 调整过程如图所示:
小根堆,Java初阶数据结构,数据结构,堆,java

你可能问会问:你是怎样得到父亲结点和他所对应的孩子结点的位置的,不就是根据完全二叉树的性质呀!

左孩子结点的下标 = (2 * 父亲结点的下标 + 1)

右孩子结点的下标 = (2 * 父亲结点的下标 + 2)

父亲结点的下标 = (孩子结点的下标 - 1) / 2

 但要注意我们的数组长度是有限的,表示孩子结点的下标不能超过数组所允许的最大下标

所以说,只要我们一开始给出要调整的二叉树的根节点坐标(父亲结点坐标),我们就能将给定的这棵完全二叉树变成小根堆;

代码示例:

/**
     * 向下调整——使得当前子树为小根堆
     * @param root 是每棵子树的根结点的下标
     * 向下调整的时间复杂度O(log2n)(最坏情况下就是树的高度)
     */
    public void shiftDown(int root) {
        int parent = root; // 父亲结点的坐标
        int child = 2 * parent + 1; // 获取左孩子结点的坐标
        // 为什么不能child下标要小于usedSize,因为当前数组的最大下标就是usedSize - 1,如果大于或等于usedSize就越界了
        while (child < usedSize) { // 每个子树在调整的时候,是按从上到下,当child的下标小于usedSize时候就结束
            // 这一步目的是找出孩子结点最大的那个值,然后在让该值和父亲结点比较(不过先要确定孩子结点存在)
            if (child + 1 < usedSize && elem[child] > elem[child + 1]) {
                child = child + 1;
            }
            if (elem[child] < elem[parent]) {
                int tmp = elem[child];
                elem[child] = elem[parent];
                elem[parent] = tmp;
                parent = child; // 从上向下调整子树,更新父亲结点的下标
                child = 2 * parent + 1; // 更新左孩子孩子结点的下标
            }
            // 因为我们是从上向下调整子树,当我们在调整上面的子树时,下面的子树一定是调整好了的,如果上面都已经满足小根堆,下面也一定满足
            else {
                break; // 此时已经是小根堆了,不需要再次调整,直接退出循环接着调整下一个子树
 
            }
        }
    }

代码解析:


3.2 小根堆的创建

说明:我们通常所用到的堆,要么是小根堆、要么就是大根堆,这里我们所创建的就一个小根堆;

既然说堆是一种可以用数组来储存,那么如果给了一个无序的数组,怎样把他变成一个有序的小根堆(完全二叉树)呢? 

我们是不是就应该在创建的过程中,及时的调整数组中的元素位置,让该堆成为一个小根堆(根结点的值大于左右孩子结点的值)

举例说明:

 小根堆,Java初阶数据结构,数据结构,堆,java

 那该怎么调整呢?

好像每一个子树都不符合小根堆的定义啊!那么我们就需要从从倒数的第一个非叶子节点的子树开始调整一直调整到根节点的树,这样从上到下调整不同的子树后就可以调整成堆;

小根堆,Java初阶数据结构,数据结构,堆,java

至于每个子树的调节方式,那不就是我们刚才提到的从根结点开始对指定的子树进行向下调节

 为什么要从下面的较小的子树调节到上面的大子树?

这是因为我们刚才的向下调整法,所能作用的二叉树是:整体上不符合堆的性质,但是其根部的左子树和右子树均满足堆的性质

当我们从上向下进行调节时,即从倒数的第一个非叶子节点的子树开始调节时,此时的二叉树一定是满足这个性质的,当下面的子树调节好了后,你会发现此时原来不满足这个性质的上面的那些子树竟然也满足这些性质了(因为他下面的子树已经调节过了)

思路如下图所示(这里是虽然构建的是大堆根,但思路是一样的): 

小根堆,Java初阶数据结构,数据结构,堆,java

 代码如下:

public class MyHeap {
    int[] elem;   // 我们说过堆可以用数组来存放
    int usedSize; // 优先级队列中有效元素的个数
    MyHeap() {
        elem = new int[10]; // 构造方法初始化一下,堆(数组)的容量
    }
    public void createHeap(int[] arrays) {
        for (int i = 0; i < arrays.length; i++) {
            elem[i] =arrays[i]; // 元素初始化,给堆中元素赋值
            ++usedSize;
        }
        // 倒数的第一个非叶子节点的子树开始调节,一直调节到根结点(根节点在数组中的下标为0)
        // 注意这里是usdSize - 1 - 1,因为父亲结点的下标 = (孩子结点的下标 - 1) / 2,
        // 我们是从倒数的第一个非叶子节点的子树开始调节的,而该子树的孩子结点坐标为usedSize - 1
        for (int i = (usedSize - 1 - 1) / 2; i >= 0 ; --i) {
            shiftDown(i); // 从下面的子树一直调到上面的子树
        }
    }
 
    /**
     * 向下调整——使得当前子树为小根堆
     * @param root 是每棵子树的根结点的下标
     * 向下调整的时间复杂度O(log2n)(最坏情况下就是树的高度)
     */
    public void shiftDown(int root) {
        int parent = root; // 父亲结点的坐标
        int child = 2 * parent + 1; // 获取左孩子结点的坐标
        // 为什么不能child下标要小于usedSize,因为当前数组的最大下标就是usedSize - 1,如果大于或等于usedSize就越界了
        while (child < usedSize) { // 每个子树在调整的时候,是按从上到下,当child的下标小于usedSize时候就结束
            // 这一步目的是找出孩子结点最大的那个值,然后在让该值和父亲结点比较(不过先要确定孩子结点存在)
            if (child + 1 < usedSize && elem[child] > elem[child + 1]) {
                child = child + 1;
            }
            if (elem[child] < elem[parent]) {
                int tmp = elem[child];
                elem[child] = elem[parent];
                elem[parent] = tmp;
                parent = child; // 从上向下调整子树,更新父亲结点的下标
                child = 2 * parent + 1; // 更新左孩子孩子结点的下标
            }
            // 因为我们是从上向下调整子树,当我们在调整上面的子树时,下面的子树一定是调整好了的,如果上面都已经满足小根堆,下面也一定满足
            else {
                break; // 此时已经是小根堆了,不需要再次调整,直接退出循环接着调整下一个子树
 
            }
        }
    }
}

3.3 向上调整算法

我们由堆的插入引入该算法

堆的插入不像先前顺序表一般,可以头插,任意位置插入等等,因为是堆,要符合大根堆或小根堆的性质,不能改变堆原本的结构,所以尾插才是最适合的,并且尾插后还要检查是否符合堆的性质。

图示说明:

小根堆,Java初阶数据结构,数据结构,堆,java

 为了确保在插入数字10后依然是个小根堆,所以要将10和28交换,依次比较父结点parent和子结点child的大小,当父小于子结点的时候,就返回,反之就一直交换,直到根部。

由前文的得知的规律,parent = (child - 1) / 2,我们可以从下到上调整子树,不断的更新child和parent下标,直到根部15

🔔需要注意的是:我们操控的是数组,但要把它想象成二叉树

画图演示调整过程:

小根堆,Java初阶数据结构,数据结构,堆,java

代码示例:

/**
     * 从下向上调整子树
     * @param child 要调整子树的孩子坐标
     */
    public void shiftUp(int child) {
        int parent = (child - 1) / 2; // 通过孩子坐标得出父亲坐标
        while (child > 0) { // 当孩子坐标等于0时,说明根结点已经调整完毕,退出循环
            // 因为之前的该子树的孩子结点的值肯定满足小根堆,所以我们只用考虑新加入的孩子结点的值
            if (elem[parent] > elem[child]) {
                int tmp = elem[parent];
                elem[parent] = elem[child];
                elem[child] = tmp;
                // 更新
                child = parent;
                parent = (child - 1) / 2;
            }
            else {
                break; // 因为我们是从下向上调整子树的,我们上面的结点之前是满足小根堆的,所以如果下面也满足了,说明整个子树都满足了,直接退出循环
            }
        }
    }

3.4 堆的插入

讲完了说完了向上调整算法,那堆的插入不就很简单了吗!

代码示例:

/**
     * 入队,当要保证入队后仍是小根堆
     * @param val 要入队的元素
     */
    public void offerHeap(int val) {
        if (isFull()) {
            elem = new int[2 * usedSize];
        }
        else {
            elem[usedSize] = val; // 把新添加的元素放到数组中最后的位置
            ++usedSize;
            shiftUp(usedSize - 1); // 放完后就从新元素的位置——从下向上调整该子树
        }
    }
 
    /**
     * 从下向上调整子树
     * @param child 要调整子树的孩子坐标
     */
    public void shiftUp(int child) {
        int parent = (child - 1) / 2; // 通过孩子坐标得出父亲坐标
        while (child > 0) { // 当孩子坐标等于0时,说明根结点已经调整完毕,退出循环
            // 因为之前的该子树的孩子结点的值肯定满足小根堆,所以我们只用考虑新加入的孩子结点的值
            if (elem[parent] > elem[child]) {
                int tmp = elem[parent];
                elem[parent] = elem[child];
                elem[child] = tmp;
                // 更新
                child = parent;
                parent = (child - 1) / 2;
            }
            else {
                break; // 因为我们是从下向上调整子树的,我们上面的结点之前是满足小根堆的,所以如果下面也满足了,说明整个子树都满足了,直接退出循环
            }
        }
    }
    
     // 判断当前队列是否已满
    public boolean isFull() {
        return usedSize == elem.length;
    }

3.4 堆的删除(堆顶元素的删除)

堆的删除和堆的插入很相似,就是把要删除的根节点和堆的最后一个元素互换位置,然后从上向下调整子树:

删除步骤:

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

举例说明:

小根堆,Java初阶数据结构,数据结构,堆,java

  删除过程如下:

小根堆,Java初阶数据结构,数据结构,堆,java

代码示例:

/**
     * 出队删除,每次删除的都是优先级高的元素——当前完全二叉树的根结点
     * 出队后仍要保证该二叉树是小根堆
     */
    public void pollHeap() {
        if (isEmpty()) {
            System.out.println("当前优先级队列为空!");
            return;
        }
        // 将当前的队首元素与队尾元素互换位置,然后将向下调整以队首元素为根节点的那个子树,使得满小根堆
        int tmp = elem[usedSize - 1];
        elem[usedSize - 1] = elem[0];
        elem[0] = tmp;
        --usedSize; // 既然是删除,有效元素个数要减一
        shiftDown(0);
    }
 
    /**
     * 向下调整——使得当前子树为小根堆
     * @param root 是每棵子树的根结点的下标
     * 向下调整的时间复杂度O(log2n)(最坏情况下就是树的高度)
     */
    public void shiftDown(int root) {
        int parent = root; // 父亲结点的坐标
        int child = 2 * parent + 1; // 获取左孩子结点的坐标
        // 为什么不能child下标要小于usedSize,因为当前数组的最大下标就是usedSize - 1,如果大于或等于usedSize就越界了
        while (child < usedSize) { // 每个子树在调整的时候,是按从上到下,当child的下标小于usedSize时候就结束
            // 这一步目的是找出孩子结点最大的那个值,然后在让该值和父亲结点比较(不过先要确定孩子结点存在)
            if (child + 1 < usedSize && elem[child] > elem[child + 1]) {
                child = child + 1;
            }
            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 boolean isEmpty() {
        return usedSize == 0;
    }

小根堆,Java初阶数据结构,数据结构,堆,java

四、优先级队列的模拟实现(小根堆)

到了这里,相信大家对小根堆的各自操作应该已经不陌生了吧!那么接下来就让我们用这些操作模拟实现一个优先级队列:

代码示例:

/**
 * 用小根堆模拟实现优先级队列
 */
public class MyHeap {
    int[] elem;
    int usedSize; // 优先级队列中有效元素的个数
    MyHeap() {
        elem = new int[10]; // 构造方法初始化一下
    }
    public void createHeap(int[] arrays) {
        for (int i = 0; i < arrays.length; i++) {
            elem[i] =arrays[i]; // 元素初始化
            ++usedSize;
        }
        for (int i = (usedSize - 1) / 2; i >= 0 ; --i) {
            shiftDown(i); // 从下面的子树一直调到上面的子树
        }
    }
 
    /**
     * 向下调整——使得当前子树为小根堆
     * @param root 是每棵子树的根结点的下标
     * 向下调整的时间复杂度O(log2n)(最坏情况下就是树的高度)
     */
    public void shiftDown(int root) {
        int parent = root; // 父亲结点的坐标
        int child = 2 * parent + 1; // 获取左孩子结点的坐标
        // 为什么不能child下标要小于usedSize,因为当前数组的最大下标就是usedSize - 1,如果大于或等于usedSize就越界了
        while (child < usedSize) { // 每个子树在调整的时候,是按从上到下,当child的下标小于usedSize时候就结束
            // 这一步目的是找出孩子结点最大的那个值,然后在让该值和父亲结点比较(不过先要确定孩子结点存在)
            if (child + 1 < usedSize && elem[child] > elem[child + 1]) {
                child = child + 1;
            }
            if (elem[child] < elem[parent]) {
                int tmp = elem[child];
                elem[child] = elem[parent];
                elem[parent] = tmp;
                parent = child; // 从上向下调整子树,更新父亲结点的下标
                child = 2 * parent + 1; // 更新左孩子孩子结点的下标
            }
            // 因为我们是从上向下调整子树,当我们在调整上面的子树时,下面的子树一定是调整好了的,如果上面都已经满足小根堆,下面也一定满足
            else {
                break; // 此时已经是小根堆了,不需要再次调整,直接退出循环接着调整下一个子树
 
            }
        }
    }
 
    /**
     * 入队,当要保证入队后仍是小根堆
     * @param val 要入队的元素
     */
    public void offerHeap(int val) {
        if (isFull()) {
            elem = new int[2 * usedSize];
        }
        else {
            elem[usedSize] = val; // 把新添加的元素放到数组中最后的位置
            ++usedSize;
            shiftUp(usedSize - 1); // 放完后就从新元素的位置——从下向上调整该子树
        }
    }
 
    /**
     * 从下向上调整子树
     * @param child 要调整子树的孩子坐标
     */
    public void shiftUp(int child) {
        int parent = (child - 1) / 2; // 通过孩子坐标得出父亲坐标
        while (child > 0) { // 当孩子坐标等于0时,说明根结点已经调整完毕,退出循环
            // 因为之前的该子树的孩子结点的值肯定满足小根堆,所以我们只用考虑新加入的孩子结点的值
            if (elem[parent] > elem[child]) {
                int tmp = elem[parent];
                elem[parent] = elem[child];
                elem[child] = tmp;
                // 更新
                child = parent;
                parent = (child - 1) / 2;
            }
            else {
                break; // 因为我们是从下向上调整子树的,我们上面的结点之前是满足小根堆的,所以如果下面也满足了,说明整个子树都满足了,直接退出循环
            }
        }
    }
    // 判断当前队列是否已满
    public boolean isFull() {
        return usedSize == elem.length;
    }
 
    /**
     * 出队删除,每次删除的都是优先级高的元素——当前完全二叉树的根结点
     * 出队后仍要保证该二叉树是小根堆
     */
    public void pollHeap() {
        if (isEmpty()) {
            System.out.println("当前优先级队列为空!");
            return;
        }
        // 将当前的队首元素与队尾元素互换位置,然后将向下调整以队首元素为根节点的那个子树,使得满小根堆
        int tmp = elem[usedSize - 1];
        elem[usedSize - 1] = elem[0];
        elem[0] = tmp;
        --usedSize; // 既然是删除,有效元素个数要减一
        shiftDown(0);
    }
    // 判断当前优先级队列是否为空
    public boolean isEmpty() {
        return usedSize == 0;
    }
 
    /**
     * 获取堆顶元素
     * @return
     */
    public int peekHeap() {
        if (isEmpty()) {
            System.out.println("当前优先级队列为空!");
            return -1;
        }
        return elem[0];
    }
}
 

总结

今天的内容就介绍到这里,我们下一节内容再见!!!!

小根堆,Java初阶数据结构,数据结构,堆,java文章来源地址https://www.toymoban.com/news/detail-729434.html

到了这里,关于【数据结构初阶】——第八节.优先级队列(小根堆的模拟实现)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

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

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

相关文章

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

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

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

    Gitee上开源的数据结构与算法代码库:数据结构与算法Gitee代码库 优先级队列,按照优先级别依次输出 计算机科学中,堆是一种基于树的数据结构,通常用 完全二叉树 实现。堆的特性如下 在大顶堆中,任意节点 C 与它的父节点 P 符合 P . v a l u e ≥ C . v a l u e P.value geq C.val

    2024年02月13日
    浏览(32)
  • 数据结构之优先级队列【堆】(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日
    浏览(42)
  • 数据结构 之 优先级队列(堆) (PriorityQueue)

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

    2024年03月20日
    浏览(34)
  • 【一起学习数据结构与算法】优先级队列(堆)

    如果我们给每个元素都分配一个数字来标记其优先级,不妨设较小的数字具有较高的优先级,这样我们就可以在一个集合中访问优先级最高的元素并对其进行查找和删除操作了。这样,我们就引入了 优先级队列 这种数据结构。 优先级队列(priority queue) 是0个或多个元素的集

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

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

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

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

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

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

    2024年02月04日
    浏览(30)
  • 【数据结构与算法】03 队列(顺序队列--循环队列--优先级队列--链队列)

    队列( queue )是一种常见的数据结构,它遵循先进先出(FIFO)的原则。队列可以理解为一个具有两个端点的线性数据结构,其中一个端点称为\\\"队尾\\\"(rear),用于插入新元素,另一个端点称为\\\"队首\\\"(front),用于移除元素。新元素被插入到队尾,而最早插入的元素总是在队

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

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

    2023年04月22日
    浏览(33)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包