本文共 2607 字,大约阅读时间需要 8 分钟。
文章收录在公众号:bigsai 关注持续分享干货和资源
事情还要从一个故事讲起:
在这里插入图片描述
在这里插入图片描述在这里插入图片描述在这里插入图片描述对于上面那只可爱的小狗狗不会,本篇即为该教程,首先,我要告诉这只可爱的小狗狗,这种问题你要使用的数据结构为优先队列,每次操作的时间复杂度为O(logn)
,而整个过程的时间复杂度为O(nlogn)
.
对于本片的设计与实现和堆排序可能有些相似,因为他们都借助堆来实现算法和数据结构,下面详细介绍优先队列的设计与实现。
而堆就是一类特殊的数据结构的统称。堆通常是一个可以被看做一棵树(完全)的数组对象。且总是满足以下规则:
对于完全二叉树,我想大家都能明白,就是最底层叶子节点要严格按照从左向右来。
在这里插入图片描述
堆有大根堆和小根堆,如果是所有父节点都大于子节点的时候,那么这就是个大根堆,反之则为小根堆,以下就是一个大根堆:在这里插入图片描述最后需要注意的是我们并不是用链式去储存这个二叉树而是用数组去存储这个树,虽然链式的使用场景可能更多一些,但是在完全二叉树的情况下空间使用率较好没有斜树的出现。并且在操作的时候可以直接通过编号找到位置进行交换。
如何理解优先队列,我们先从一段对话说起:
在这里插入图片描述
优先队列,它是一个队列。而普通的队列遵从先进先出的规则。而优先队列遵循一个排序的规则:每次抛出自定义排序最大(小)的,默认的情况是抛出最小的,本篇也就从最基本的原理进行分析。
并且它的用法队列还是一样的,,所以我们在设计这个类的时候api方面要与队列的api一致。
在这里插入图片描述
我们主要实现add、poll、和peek方法,并且会着重于算法的实现而不太着重一些细节的实现。
虽然优先队列和堆排序利用堆结构特性的流程有一些相似,但是两者其实还是有些操作上的区别的:
堆排序 :
优先队列:
但是优先队列的具体操作流程是如何的呢?我们具体分析其插入和删除的流程。
插入add流程(小根堆为例):
在这里插入图片描述
删除pop流程(小根堆为例):
在这里插入图片描述
我想到这里,优先队列的内部流程思想你已经掌握了,但是懂优先队列原理和手写优先队列是两个概念,为了更深入的学习优先队列,在这里就带你手写一个简易型的优先队列。
在代码的具体实现方面,最主要的就是pop()和add()两个函数了。在pop()函数具体实现的时候,将最后一个元素移到堆头考虑和其他孩子节点交换的时候,用while进行操作的时候计算孩子下标的时候要确保不越界。我们用的是数组存储数据,优先队列的长度不一定等于这个数组的长度。
而在实现add()函数的时候,这里简单的考虑了一下扩容。
具体实现的代码为:
import java.util.Arrays;public class priQueue { private int size;//优先队列的大小 private int capacity;//数组的容量 private int value[];//储存的值 public priQueue() { this.capacity = 10; this.value = new int[capacity]; this.size=0; } public priQueue(int capacity) { this.capacity = capacity; this.value = new int[capacity]; this.size=0; } /** * 插入元素 * @param number */ public void add(int number) { if(size==capacity)//扩容 { capacity*=1.5; value= Arrays.copyOf(value,capacity); } value[size++]=number;//先加到末尾 int index=size-1; while (index>=1) {//进行交换 if(value[index]=size)//左孩子必须满足在条件内 break; else if(rightChild
写个类测试一下看看:
在这里插入图片描述
在这里插入图片描述
本次优先队列介绍就到这里啦,感觉不错记得点赞或一键三连哦,建议和堆排序一起看和学习效果更佳,要能够手写代码。个人公众号:bigsai
回复 bigsai 更多精彩和资源与你分享。
在这里插入图片描述
转载地址:http://pvvkz.baihongyu.com/