数据结构——优先队列与堆

简介: 数据结构——优先队列与堆

一、什么是优先队列与堆



优先级队列的底层实际上就是利用堆这种数据结构实现的。


堆在逻辑上是一棵完全二叉树,而其物理结构却是数组,二叉树中的每个节点对应着数组的下标树中的第一个节点(根节点)的数组下标为0,树中的最后一个节点不一定为数组的最后一个元素(数组可能没满),而我们学习的堆,分为两种:


① 大顶堆:每个双亲节点的值都 >= 其左右孩子节点的值,此外,根节点的值一定是所有堆中其余节点的最大值。


81e874a36c5d49d8a03ef3b6fd61e8cf.png


② 小顶堆:每个双亲节点的值都 <= 其左右孩子节点的值,此外,根节点的值一定是所有堆中其余节点的最小值。


764ac31375a74f40a5fc3b83e9c9835d.png


② 小顶堆:每个双亲节点的值都 <= 其左右孩子节点的值,此外,根节点的值一定是所有堆中其余节点的最小值。


764ac31375a74f40a5fc3b83e9c9835d.png


二、通过代码创建一个堆



说明一下:这里只是采用个人逻辑创建的一个堆,旨在与更深入地理解堆这个数据结构。


创建一个 .java 文件,里面放着一个类 Heap,通过 createHeap( ) 方法和 shiftDown( ) 来模拟一个堆。


public class Heap {
    public int [] elem;
    public int usedSize;
    public Heap() {
        this.elem = new int[10];
    }
    //创建一个堆
    public void createHeap(int[] arr){
        for (int i = 0; i < arr.length; i++) {
            this.elem[i] = arr[i];
            usedSize++;
        }
        for (int parent = ((usedSize-1)-1) / 2; parent >= 0 ; parent--) {
            shiftDown(parent, usedSize);
        }
    }
    //向下调整
    public void shiftDown(int parent, int len){
        int child = parent*2 +1 ;
        while (child < len){
            if(child+1 < len && elem[child] < elem[child+1]){
                child++; //保证 child 走到左右最大的一个节点
            }
            if(elem[parent] < elem[child]){
                int temp = elem[child];
                elem[child] = elem[parent];
                elem[parent] = temp;
            }else {
                break;
            }
            parent = child;
            child = parent*2 +1 ;
        }
    }
    //打印堆元素(层序)
    public void print(){
        for (int i = 0; i < usedSize; i++) {
            System.out.print(elem[i] + " ");
        }
        System.out.println();
    }
}


创建另一个 .java 文件,里面用主函数进行测试


public class Test {
    public static void main(String[] args) {
        int[] arr = {4,0,2,1,5,6,9,8,3,7};
        Heap heap = new Heap();
        System.out.println(Arrays.toString(arr));
        heap.createHeap(arr);
        System.out.println("-------------大顶堆-------------");
        heap.print();
    }
}


原始的堆:


91a55ac8805e44a9a6468cf3f61eb16d.png


优化成大顶堆:


777c8de5716c4c87ba4f90b93c9869b5.png


输出结果:


281cc009ab394f5ea3ae2bee9274b06c.png


在上述代码中,如果我们创建一个小顶堆,那么只需要更改几个不等号即可。


三、堆排序



一类问题:给定一组数据,将其按升序/逆序排序。


我们见过冒泡排序,但是本次我们通过堆排序进行操作。


举例,给定下列的数组 arr,让其升序排序


int[] arr = {4,0,2,1,5,6,9,8,3,7};
//排序后:0 1 2 3 4 5 6 7 8 9


堆排序思路:


① 在排序数组之前,我们要让数组符合大顶堆的逻辑

② 循环交换堆顶元素(即数组第一个元素)和堆尾元素(即数组最后一个元素),那么这样一来,数组的末尾逐渐排好


注意:每交换一次元素,我们就得重新让堆变成大顶堆逻辑,之后才能继续交换,这样做目的是:我们始终要让堆顶的元素处于最大值。


创建一个 .java 文件,里面放着一个类 Heap,通过 createHeap( ) 方法和 shiftDown( ) 来模拟一个堆,通过 heapSort( ) 进行堆排序


public class Heap {
    public int [] elem;
    public int usedSize;
    public Heap() {
        this.elem = new int[10];
    }
    //创建一个大顶堆
    public void createHeap(int[] arr){
        for (int i = 0; i < arr.length; i++) {
            this.elem[i] = arr[i];
            usedSize++;
        }
        for (int parent = ((usedSize-1)-1) / 2; parent >= 0 ; parent--) {
            shiftDown(parent, usedSize);
        }
    }
    //向下调整
    public void shiftDown(int parent, int len){
        int child = parent*2 +1 ;
        while (child < len){
            if(child+1 < len && elem[child] < elem[child+1]){
                child++; //保证 child 走到左右最大的一个节点
            }
            if(elem[parent] < elem[child]){
                int temp = elem[child];
                elem[child] = elem[parent];
                elem[parent] = temp;
            }else {
                break;
            }
            parent = child;
            child = parent*2 +1 ;
        }
    }
    public void heapSort(){
        int end = usedSize-1;
        while(end > 0){
            int temp = elem[0];
            elem[0] = elem[end];
            elem[end] = temp;
            end--;
            shiftDown(0,end);
        }
    }
    //打印堆元素(层序)
    public void print(){
        for (int i = 0; i < usedSize; i++) {
            System.out.print(elem[i] + " ");
        }
        System.out.println();
    }
}


创建另一个 .java 文件,里面用主函数进行测试


public class Test {
    public static void main(String[] args) {
        int[] arr = {4,0,2,1,5,6,9,8,3,7};
        Heap heap = new Heap();
        System.out.println(Arrays.toString(arr));
        heap.createHeap(arr);
        System.out.println("-------------创建一个大顶堆-------------");
        heap.print();
        System.out.println("-------------进行堆排序-------------");
        heap.heapSort();
        heap.print();
    }
}


输出结果:


7f1d0c83d0334538bc722305bac6e259.png


四、Java 实现的优先队列



我们来测试一下 Java 实现的优先级队列,并使用一些对应的方法:


可以发现 Java 集合框架中实现的优先级队列,它的底层实际上默认就是一个小顶堆。而我们不管使用了 offer( ) 方法,还是 poll( ) 方法,使用过后,Java 底层会自动将数组变换成小顶堆,也就是说:只要你向优先级队列中传入了数据,那么之后你所进行的所有操作,都只应用于小顶堆。


import java.util.PriorityQueue;
public class Test1 {
    public static void main(String[] args) {
        PriorityQueue<Integer> priorityQueue = new PriorityQueue<>();
        priorityQueue.offer(7);
        System.out.println(priorityQueue);
        priorityQueue.offer(3);
        System.out.println(priorityQueue);
        priorityQueue.offer(1);
        System.out.println(priorityQueue);
        priorityQueue.offer(5);
        System.out.println(priorityQueue);
        System.out.println(priorityQueue.size());
        System.out.println(priorityQueue.peek());
        System.out.println(priorityQueue.poll());
        System.out.println(priorityQueue);
        System.out.println(priorityQueue.poll());
        System.out.println(priorityQueue);
        System.out.println(priorityQueue.size());
    }
}


输出结果:


dd63360ef46c4521ae2794b9f4e129a9.png


1. 深度解析offer( ) 方法


这里的 offer( ) 方法采用了向上调整的逻辑,与我们之前的建堆逻辑不同,因为这里是从数组的尾部一个一个加进去的,通俗地说,就是先尾插,然后 Java 实现的优先级队列会自动为我们调整为小顶堆。


e7da254a0e624524ad1cc4a8bc7f764b.png


2. 深度解析poll( ) 方法


这里的 poll ( ) 方法并不是从数组的头部一个一个删除的,而是先将数组的第一个元素与数组的最后一个元素进行交换,之后进行调整。也就是说:我们每一次 poll 之后,底层就会帮我们调整为小顶堆。


225c48b311c44de3ba8e4f5b884947a7.png


下面是进行了优先队列的循环添加和循环删除,更好理解:


import java.util.PriorityQueue;
public class Test2 {
    public static void main(String[] args) {
        PriorityQueue<Integer> priorityQueue = new PriorityQueue<>();
        int[] arr = {4,2,5,6,0,9,1,8,3,7};
        for (int i = 0; i < 10; i++) {
            priorityQueue.offer(arr[i]);
            System.out.println(priorityQueue);
        }
        System.out.println("----------------------------------");
        for (int i = 0; i < 10; i++) {
            priorityQueue.poll();
            System.out.println(priorityQueue);
        }
    }
}


输出结果:


df05442d6f6b49f3a3e29dd48b32e6e6.png


五、深度理解 Java 优先队列底层的代码



import java.util.Arrays;
import java.util.Comparator;
import java.util.PriorityQueue;
public class Test1 {
    public static void main(String[] args) {
        //默认 优先级队列中的元素为11
        PriorityQueue<Integer> minHeap = new PriorityQueue<>(new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o1 - o2;
            }
        });
        int [] arr1 = {4,0,2,1,5,6,9,8,3,7};
        System.out.println(Arrays.toString(arr1));
        for (int i = 0; i < arr1.length; i++) {
            minHeap.offer(arr1[i]);
        }
        System.out.println("---------------小顶堆---------------");
        System.out.println(minHeap);
        PriorityQueue<Integer> maxHeap = new PriorityQueue<>(new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o2 - o1;
            }
        });
        for (int i = 0; i < arr1.length; i++) {
            maxHeap.offer(arr1[i]);
        }
        System.out.println("---------------大顶堆---------------");
        System.out.println(maxHeap);
        int k = 3;
        PriorityQueue<Integer> maxHeap2 = new PriorityQueue<>(k,new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o2 - o1;
            }
        });
        for (int i = 0; i < k; i++) {
            maxHeap2.offer(arr1[i]);
        }
        System.out.println("---------------大顶堆---------------");
        System.out.println(maxHeap2);
    }
}


输出结果如下,关于使用 offer( ) 方法后,怎么进行调整为大顶堆/小顶堆,我已经通过上面两幅图说明的很清楚了。接下来我们一起来看看为什么我们要让堆实现内部类。


输出结果:


b2cf2cd246e145488ab3114e4b3698ff.png


总结:


下面的这幅图是 Java 底层实现优先级队列的源码,当我们只看代码的时候,我要说明以下几点


① 在 PriorityQueue 类,当我们没有给定初始容量时,其默认容量为 11 ,当然我们也可以自己给定初始容量。

② 当我们不实现内部类,不传参的时候, PriorityQueue 类默认底层实现的是小顶堆。

③ 我们可以通过接口 Comparator,来重写 compare( ) 方法,以此来实现小顶堆或大顶堆。


a2ab672df11a45ce83203536844f5e51.png


小顶堆和大顶堆的语法格式


格式1


//默认为小顶堆,初始优先级队列的数组容量为11
PriorityQueue<Integer> minHeap = new PriorityQueue<>();


格式2


为什么会出现格式2 这种形式?


在下面的代码块中,实际上我们并不是直接通过 Comparator 接口 new 了一个对象,而是通过匿名内部类实现了 Comparator 接口,并重写了 compare( ) 方法,这就是匿名内部类的语法。而这么做是为了更清楚地知道优先队列底层:堆这个数据结构是按什么比较的。


//小顶堆,初始优先级队列的数组容量为11
PriorityQueue<Integer> minHeap = new PriorityQueue<>(new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o1 - o2;
            }
        });


格式3


在下面的格式中,我们可以看到,在 compare( ) 方法中,我们将 o2 和 o1 的位置调换过来了,对照格式2,我们就能清楚地明白,下面地程序是一个大顶堆地构造方式,因为大顶堆的堆顶永远是最大的元素。


//大顶堆,初始容量为3
PriorityQueue<Integer> maxHeap = new PriorityQueue<>(3,new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o2 - o1;
            }
        });


六、使用堆解决TopK问题



一类问题:给你一组数据,让你求这组数据中前 k 个最大的 / 最小的元素


举例题目:给你下面的一个数组 arr,求出其前三个最小的元素


int[] arr = {4,2,5,6,0,9,1,8,3,7};
//最小的三个元素 0,1,2


下面两个方法求的均是前 k 个最小的元素,当求前 k 个最大的元素逻辑其实是一样的,只是转换一下堆的使用即可。


方法一


  1. 创建一个大小为 3 的一个大顶堆
  2. 开始遍历数组
    (1)前 k 个元素组建成大顶堆
    (2)从 k+1 个元素开始,开始进行与堆顶判断
    (3)如果新添加的元素 < 堆顶的元素,那么就先头删,再尾插,过程中底层会自动帮我们调整大顶堆顺序
  3. 把前 k 个元素放入一个新的数组中,并返回给主函数


public class Test1 {
    public static void main(String[] args) {
        int[] arr = {4,2,5,6,0,9,1,8,3,7};
        int k = 3;
        int[] ret = topK(arr,k);
        System.out.println(Arrays.toString(ret));
    }
    public static int[] topK(int[] arr, int k){
        //1. 创建一个大小为 k 的一个大顶堆
        PriorityQueue<Integer> maxHeap = new PriorityQueue<>(k, new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o2 - o1;
            }
        });
        //2. 开始遍历数组
        for (int i = 0; i < arr.length; i++) {
            //2.1 前 k 个元素组建成大顶堆
            if(maxHeap.size() < k){
                maxHeap.offer(arr[i]);
            }else {
                //2.2 从 k+1 个元素开始,开始进行与堆顶判断
                int top = maxHeap.peek();
                if(top > arr[i]){
                    maxHeap.poll();
                    maxHeap.offer(arr[i]);
                }
            }
        }
        //3. 把前 k 个元素放入一个新的数组中,并返回给主函数
        int [] ret = new int[k];
        for (int i = 0; i < k; i++) {
            ret[i] = maxHeap.poll();
        }
        return ret;
    }
}


输出结果:


ec64ab3f90724a58b4439b4731fd79fb.png


方法二


  1. 创建一个小顶堆
  2. 将堆顶元素一个一个弹出至 ret 数组中


import java.util.Arrays;
import java.util.PriorityQueue;
public class Test1 {
    public static void main(String[] args) {
        int[] arr = {4,2,5,6,0,9,1,8,3,7};
        int k = 3;
        topK(arr,k);
    }
    public static void topK(int[] arr, int k){
        //1. 创建一个小顶堆
        PriorityQueue<Integer> minHeap = new PriorityQueue<>();
        for (int i = 0; i < arr.length; i++) {
            minHeap.offer(arr[i]);
        }
        System.out.println(minHeap);
        int [] ret = new int[k];
        //2. 将堆顶元素弹出至 ret 数组中
        for (int i = 0; i < k; i++) {
            ret[i] = minHeap.poll();
        }
        System.out.println(Arrays.toString(ret));
    }
}


输出结果:


44313fbe6ec7411eb9e62bf1c626bcfc.png



七、leetcode 373 查找和最小的K对数字



leetcode 373


下面的两种方法就是使用我刚刚堆的性质来做的,主要我们需要理解一下思想


方法一


class Solution {
    public List<List<Integer>> kSmallestPairs(int[] nums1, int[] nums2, int k) {
        //1. 创建一个大顶堆
        PriorityQueue<List<Integer>> maxHeap = new PriorityQueue<>(k, new Comparator<List<Integer>>() {
            @Override
            public int compare(List<Integer> o1, List<Integer> o2) {
                return ( o2.get(0)+o2.get(1) ) - ( o1.get(0)+ o1.get(1) ) ;
            }
        });
        //2. 遍历二维数组
        for (int i = 0; i < Math.min(nums1.length,k); i++) {
            for (int j = 0; j < Math.min(nums2.length,k); j++) {
                //2.1 将前三个一维数组放入大顶堆中
                if(maxHeap.size() < k){
                    List<Integer> list = new ArrayList<>();
                    list.add(nums1[i]);
                    list.add(nums2[j]);
                    maxHeap.offer(list);
                }else{
                    //2.2 从第四个一维数组开始,与堆顶数组进行判断
                    int top = maxHeap.peek().get(0) + maxHeap.peek().get(1);
                    if( nums1[i] + nums2[j] < top){
                        maxHeap.poll();
                        List<Integer> list = new ArrayList<>();
                        list.add(nums1[i]);
                        list.add(nums2[j]);
                        maxHeap.offer(list);
                    }
                }
            }
        }
        List<List<Integer>> ret  = new ArrayList<>();
        for (int i = 0; i < k && !maxHeap.isEmpty() ; i++) {
            ret.add(maxHeap.poll());
        }
        return ret;
    }
}


为什么 for 循环的边界可以使用 Math.min( length, k) ?因为题目给出的原数组是两个升序的数组。


for (int i = 0; i < Math.min(nums1.length,k); i++) {
  for (int j = 0; j < Math.min(nums2.length,k); j++) {
  }
}



方法二


class Solution {
    public List<List<Integer>> kSmallestPairs(int[] nums1, int[] nums2, int k) {
        PriorityQueue<List<Integer>> minHeap = new PriorityQueue<>(k, new Comparator<List<Integer>>() {
            @Override
            public int compare(List<Integer> o1, List<Integer> o2) {
                return (o1.get(0)+o1.get(1)) - (o2.get(0)+o2.get(1));
            }
        });
        for (int i = 0; i < Math.min(nums1.length,k); i++) {
            for (int j = 0; j < Math.min(nums2.length,k); j++) {
                List<Integer> list = new ArrayList<>();
                list.add(nums1[i]);
                list.add(nums2[j]);
                minHeap.offer(list);
            }
        }
        List<List<Integer>> ret = new ArrayList<>();
        for (int i = 0; i < k && !minHeap.isEmpty(); i++) {
            ret.add(minHeap.poll());
        }
        return ret;
    }
}


目录
相关文章
|
2月前
|
存储 算法 Java
散列表的数据结构以及对象在JVM堆中的存储过程
本文介绍了散列表的基本概念及其在JVM中的应用,详细讲解了散列表的结构、对象存储过程、Hashtable的扩容机制及与HashMap的区别。通过实例和图解,帮助读者理解散列表的工作原理和优化策略。
51 1
散列表的数据结构以及对象在JVM堆中的存储过程
|
2月前
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
126 16
|
3月前
|
存储 JavaScript 前端开发
为什么基础数据类型存放在栈中,而引用数据类型存放在堆中?
为什么基础数据类型存放在栈中,而引用数据类型存放在堆中?
144 1
|
4月前
|
存储 Java
【数据结构】优先级队列(堆)从实现到应用详解
本文介绍了优先级队列的概念及其底层数据结构——堆。优先级队列根据元素的优先级而非插入顺序进行出队操作。JDK1.8中的`PriorityQueue`使用堆实现,堆分为大根堆和小根堆。大根堆中每个节点的值都不小于其子节点的值,小根堆则相反。文章详细讲解了如何通过数组模拟实现堆,并提供了创建、插入、删除以及获取堆顶元素的具体步骤。此外,还介绍了堆排序及解决Top K问题的应用,并展示了Java中`PriorityQueue`的基本用法和注意事项。
76 5
【数据结构】优先级队列(堆)从实现到应用详解
|
3月前
|
存储 算法 调度
数据结构--二叉树的顺序实现(堆实现)
数据结构--二叉树的顺序实现(堆实现)
|
3月前
|
存储 算法 分布式数据库
【初阶数据结构】理解堆的特性与应用:深入探索完全二叉树的独特魅力
【初阶数据结构】理解堆的特性与应用:深入探索完全二叉树的独特魅力
|
3月前
|
存储 算法
探索数据结构:分支的世界之二叉树与堆
探索数据结构:分支的世界之二叉树与堆
|
3月前
|
存储 算法 Java
【用Java学习数据结构系列】用堆实现优先级队列
【用Java学习数据结构系列】用堆实现优先级队列
47 0
|
3月前
|
存储 算法
【数据结构】二叉树——顺序结构——堆及其实现
【数据结构】二叉树——顺序结构——堆及其实现
|
3月前
【数据结构】大根堆和小根堆
【数据结构】大根堆和小根堆
82 0

热门文章

最新文章