什么是Java优先级队列?
PriorityQueue是基于无界优先级队列和优先级堆构建的重要Java API之一。本文通过适当的代码示例深入了解了有关此API及其用法的一些复杂信息。另在上篇文章中我们简单地谈了下Java编译器API简介,你可以先阅读以方便更好地阅读这篇文章。感谢优锐课老师对我写下这两篇文章时给予的帮助。
总览
PriorityQueue类是java.util包的一部分,是Java中基于优先级的队列的通用实现。队列基本上是一种数据结构,它定义了针对商店中商品的插入和检索过程的特定规范。这个想法与排队等候说票的许多人非常相似。排队的第一个人有第一个获得彩票的机会,最后一个人有一个结局的机会。人们被添加到队列的末尾或末尾。在队列中添加项目在技术上称为入队过程,从队列中删除的项目是从该行的第一行开始。这称为出队。想法是以先进先出的方式对元素进行排序。
现在,这是最简单的体系结构,并且紧密定义了队列的实际含义以及如何在计算机中模拟队列。存储通常由一个简单的数组表示,其中存储和检索过程具有此定义的规范。优先级队列在此之上强加一些特殊规范。我们将在后面看到更多内容。
队列的Java实现
Java API在java.util包中具有通用接口名称Queue <E>。这是Java Collection Framework API的一部分,旨在在处理之前保存元素。作为集合的一部分,它具有所有基本的集合操作。特定于其标识的操作涉及存储在其中的元素的插入,提取和检查。这些操作中的每一个都有两种不同的形式,例如一种在操作失败时引发异常,而另一种则根据操作返回一个特殊值,例如null或false。请注意,与典型队列不同,Java Queue的具体实现不必一定以FIFO方式对元素进行排序。对于基于优先级的队列尤其如此,其中元素的排序是根据提供的比较器或自然排序完成的。但是无论顺序如何,remove()或poll()方法将始终检索队列开头的元素。这两种不太可能的方法之间的特定区别似乎是一种相似的方法,即在失败时引发异常(NoSuchElementException),而后者则返回特殊值(null)。
方法 |
引发异常 |
描述 |
|||
E remove() |
NoSuchElementException |
从队列的开头检索一个元素。 |
|||
void add(E) |
IllegalStateException |
在队列末尾插入一个元素。成功返回true,如果空间不可用则抛出异常。 |
|||
E element() |
NoSuchElementException |
检索元素而不将其从队列的开头移除。 |
|||
方法 |
返回特殊值 |
说明 |
|||
boolean offer(E) |
true or false |
将元素插入队列。成功返回true,如果由于空间不足而无法插入,则返回false。 |
|||
E poll() |
null |
从队列的开头移除元素;如果队列为空,则返回null。 |
|||
E peek() |
null |
检索但不从队列的开头删除元素。如果队列为空,则返回null。 |
|||
请注意,Queue <E>接口不适用于并发编程,因为它没有定义阻塞队列方法,在这种方法中,入队和出队过程等待元素出现在队列中或大小可用。有一个名为BlockingQueue <E>的特定接口,该接口扩展了Queue <E>接口并解决了这些问题。
有一个称为AbstractQueue <E>的抽象类,该类提供某些队列操作的部分实现。 PriorityQueue <E>类是此抽象类的直接扩展。
优先队列
优先级队列的Java实现是一种特殊的队列,其中元素的排序由其自然排序原则确定,也可以根据创建期间提供的Comparator进行定制。我们在构造过程中调用的构造函数决定要与优先级队列一起使用的排序原则。与不允许使用null元素的Queue <E>不同,但是某些实现(例如LinkedList)也不禁止插入null元素。但是,PriorityQueue <E>根本不允许空元素。如果优先级队列是根据自然顺序构造的,则任何不可比较的元素插入都将引发ClassCastException。
它被声明为无限制的并且基于优先级堆。尽管队列的大小被称为无限制,但内部具有确定阵列大小的能力。插入元素时,此大小会自动增长。但是,没有详细说明增大尺寸的原理。
有七种类型的重载构造函数,通过它们我们可以设置参数来指定队列的初始容量,提供Comparator来指定元素的自定义顺序,或者使用无参数构造函数将所有内容接受为默认值。
- PriorityQueue()
- PriorityQueue(int initialCapacity)
- PriorityQueue(int initialCapacity, Comparator<? Super E> comparator)
- PriorityQueue(Commection<? extends E> c)
- PriorityQueue(Comparator<? Super E> comparator)
- PriorityQueue(PriorityQueue<? extends E> c)
- PriorityQueue(SortedSet<? extends E> c)
与Queue <E>相似,PriorityQueue <E>也不同步,因此在并发编程中应谨慎使用。但是,有一个同步的替代方法,称为PriorityBlockingQueue <E>。这与PriorityQueue <E>的工作方式相同,只是具有线程安全的其他限定条件。
PriorityQueue <E>中定义的操作与Queue <E>相同,但有一些附加功能。
方法 |
描述 |
void clear() |
从优先级队列中删除所有元素。 |
Comparator<? Super E> comparator() |
返回与队列关联的比较器。如果根据自然顺序对队列进行排序,则返回null。 |
boolean contains(Object o) |
如果队列包含指定的对象,则返回true。 |
Iterator<E> iterator() |
返回与Collection类关联的旧式迭代器。但是,它不能保证以任何特定顺序遍历元素。 |
Spliterator<E> spliterator() |
创建后期绑定,故障快速拆分器,但具有与迭代器相同的条件。 |
Object[] toArray() |
这是一种便捷方法,可用于设置遍历顺序权限,例如Arrays.sort(priorityQueue.toArray()). |
<T>T[] toArray(T[] a) |
返回数组元素,但是返回类型由指定的数组确定。 |
快速示例1
让我们用一个简单的程序实现PriorityQueue <E>的一些操作。
package org.mano.examples;
import java.util.Arrays;
import java.util.Iterator;
import java.util.PriorityQueue;
public class Example1 {
public static void main(String[] args){
PriorityQueue<String> pq = new PriorityQueue<>();
pq.add("Mercury");
pq.add("Venus");
pq.add("Earth");
pq.add("Mars");
pq.add("Jupiter");
pq.add("Saturn");
// Get the most priority element based upon
// natural alphabetic ordering in string
System.out.println("Priority element "+pq.peek());
// Queue elements
show(pq);
// Remove top of the queue element
pq.poll();
show(pq);
// Retrieves element from the head of the queue
pq.remove("Earth");
show(pq);
String result = pq.contains("Earth")?
"Found Earth":"Earth Missing!";
System.out.println(result);
Object[] arr = pq.toArray();
Arrays.sort(arr);
System.out.println("");
for (int i = 0; i<arr.length; i++)
System.out.print(arr[i].toString()+"::");
}
public static void show(PriorityQueue<String> pq){
Iterator<String> itr = pq.iterator();
while (itr.hasNext())
System.out.print(itr.next()+"::");
System.out.println("");
}
}
Output
Priority element Earth
Earth::Jupiter::Mercury::Venus::Mars::Saturn::
Jupiter::Mars::Mercury::Venus::Saturn::
Jupiter::Mars::Mercury::Venus::Saturn::
Earth Missing! Jupiter::Mars::Mercury::Saturn::Venus::
快速示例2
这是另一个带有自定义比较器的快速示例。
package org.mano.examples;
import java.util.Comparator;
import java.util.PriorityQueue;
public class Planet implements Comparable<Planet>{
private String name;
private double orbitPeriodInDays;
public Planet(String name, double orbitPeriodInDays) {
this.name = name;
this.orbitPeriodInDays = orbitPeriodInDays;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public double getOrbitPeriodInDays() {
return orbitPeriodInDays;
}
public void setOrbitPeriodInDays(double orbitPeriodInDays) {
this.orbitPeriodInDays = orbitPeriodInDays;
}
@Override
public int compareTo(Planet o) {
return 0;
}
@Override
public String toString() {
return "Planet{" +
"name='" + name + '\'' +
", orbitPeriodInDays=" + orbitPeriodInDays +
'}';
}
public static void main(String[] args){
Comparator<Planet> nameSorter =
Comparator.comparing(Planet::getName);
PriorityQueue<Planet> priorityQueue = new
PriorityQueue<>(nameSorter);
priorityQueue.add(new Planet("Mercury",88));
priorityQueue.add(new Planet("Venus",225));
priorityQueue.add(new Planet("Earth",365.24));
priorityQueue.add(new Planet("Mars",693.5));
priorityQueue.add(new Planet("Jupiter",4343.5));
priorityQueue.add(new Planet("Saturn",10767.5));
priorityQueue.add(new Planet("Uranus",30660));
priorityQueue.add(new Planet("Neptune",60152));
Object[] list = priorityQueue.toArray();
for (Object o: list)
System.out.println(o);
}
}
Output
Planet{name='Earth', orbitPeriodInDays=365.24}
Planet{name='Jupiter', orbitPeriodInDays=4343.5}
Planet{name='Mercury', orbitPeriodInDays=88.0}
Planet{name='Neptune', orbitPeriodInDays=60152.0}
Planet{name='Mars', orbitPeriodInDays=693.5}
Planet{name='Saturn', orbitPeriodInDays=10767.5}
Planet{name='Uranus', orbitPeriodInDays=30660.0}
Planet{name='Venus', orbitPeriodInDays=225.0}
总结
优先级队列的其他规范是,从列表中删除的项目具有最高优先级。Java将优先级规则强加给其他常规队列的方式是通过附加元素的排序原则。该顺序可以根据程序员的要求进行自定义,也可以设置为默认。这就是Java中优先级队列实现的本质。
感谢阅读!
什么是Java优先级队列?的更多相关文章
- Java优先级队列
package com.lk.A; import java.util.PriorityQueue; public class Test5 { public static void main(Strin ...
- JAVA优先级队列元素输出顺序测试
package code.test; import java.util.Comparator; import java.util.Iterator; import java.util.Priority ...
- Java优先级队列实现
优先级队列数组实现: public class PriorityQueue { private int[] data; private int size; public PriorityQueue(i ...
- 什么是Java优先级队列(Priority Queue)?
PriorityQueue是一个基于优先级堆的无界队列.它的元素是按照自然顺序排序的.在创建元素的时候,我们给它一个一个负责排序的比较器.PriorityQueue不允许null值,因为 它们没有自然 ...
- java优先级队列的使用 leecode.703.数据流中的第K大元素
//设计一个找到数据流中第K大元素的类(class). //注意是排序后的第K大元素,不是第K个不同的元素. class KthLargest { private PriorityQueue<I ...
- Java中的队列Queue,优先级队列PriorityQueue
队列Queue 在java5中新增加了java.util.Queue接口,用以支持队列的常见操作.该接口扩展了java.util.Collection接口. Queue使用时要尽量避免Collecti ...
- java中PriorityQueue优先级队列使用方法
优先级队列是不同于先进先出队列的另一种队列.每次从队列中取出的是具有最高优先权的元素. PriorityQueue是从JDK1.5开始提供的新的数据结构接口. 如果不提供Comparator的话,优先 ...
- java PriorityBlockingQueue 基于优先级队列,的读出操作可以阻止.
java PriorityBlockingQueue 基于优先级队列.的读出操作可以阻止. package org.rui.thread.newc; import java.util.ArrayLis ...
- 《转》JAVA中PriorityQueue优先级队列使用方法
该文章转自:http://blog.csdn.net/hiphopmattshi/article/details/7334487 优先级队列是不同于先进先出队列的另一种队列.每次从队列中取出的是具有最 ...
随机推荐
- 基于crypto++国产加密软件SM4的实现,顺带加了ase,base64
唔,美国压制得越狠,各种替代产品就越能活. 本文分享SM4的一种快速实现与集成方式. SM4(原名SMS4)是中华人民共和国政府采用的一种分组密码标准,由国家密码管理局于201 ...
- 打算写一个《重学Node.js》系列,希望大家多多支持
先放上链接吧,项目已经开始2周了:https://github.com/hellozhangran/happy-egg-server 想法 现在是2019年11月24日,还有人要开始学习Node.js ...
- php+redis实现注册、删除、编辑、分页、登录、关注等功能
本文实例讲述了php+redis实现注册.删除.编辑.分页.登录.关注等功能.分享给大家供大家参考,具体如下: 主要界面 连接redis redis.php <?php //实例化 $red ...
- vue e.path 移动端兼容
作用 e.path 用来获取点击元素及以上所有父元素的一个数组 问题 当在移动端会有获取不到e.path的问题 不兼容 解决 let path = event.path || (event.compo ...
- 【Android - 自定义View】之View的draw过程解析
draw(绘制)过程的作用是将View绘制到屏幕上面.View中有 draw() 方法和 onDraw() 方法,但onDraw()方法是空方法:ViewGroup中没有draw()方法,也没有onD ...
- Machine Learning In Action 第二章学习笔记: kNN算法
本文主要记录<Machine Learning In Action>中第二章的内容.书中以两个具体实例来介绍kNN(k nearest neighbors),分别是: 约会对象预测 手写数 ...
- 使用 sroll-snap-type 优化滚动
根据 CSS Scroll Snap Module Level 1 规范,CSS 新增了一批能够控制滚动的属性,让滚动能够在仅仅通过 CSS 的控制下,得到许多原本需要 JS 脚本介入才能实现的美好交 ...
- Chapter 07-Basic statistics(Part2 Frequency and contingency tables)
这一部分使用在vcd包中的Arthritis数据集. > library(vcd) 载入需要的程辑包:MASS 载入需要的程辑包:grid 载入需要的程辑包:colorspace > he ...
- 计算购物车金额总和( jquery )
今天简单写了一个jq版购物车计算金额总和的例子,如图: 整体页面代码如下: <!DOCTYPE html> <html> <head> <meta chars ...
- c++ use curllib send REST API request to Web server
E.g and explaination: Pre-condition: Server is running and can reciever CURL command with json forma ...