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优先级队列?的更多相关文章

  1. Java优先级队列

    package com.lk.A; import java.util.PriorityQueue; public class Test5 { public static void main(Strin ...

  2. JAVA优先级队列元素输出顺序测试

    package code.test; import java.util.Comparator; import java.util.Iterator; import java.util.Priority ...

  3. Java优先级队列实现

    优先级队列数组实现: public class PriorityQueue { private int[] data; private int size; public PriorityQueue(i ...

  4. 什么是Java优先级队列(Priority Queue)?

    PriorityQueue是一个基于优先级堆的无界队列.它的元素是按照自然顺序排序的.在创建元素的时候,我们给它一个一个负责排序的比较器.PriorityQueue不允许null值,因为 它们没有自然 ...

  5. java优先级队列的使用 leecode.703.数据流中的第K大元素

    //设计一个找到数据流中第K大元素的类(class). //注意是排序后的第K大元素,不是第K个不同的元素. class KthLargest { private PriorityQueue<I ...

  6. Java中的队列Queue,优先级队列PriorityQueue

    队列Queue 在java5中新增加了java.util.Queue接口,用以支持队列的常见操作.该接口扩展了java.util.Collection接口. Queue使用时要尽量避免Collecti ...

  7. java中PriorityQueue优先级队列使用方法

    优先级队列是不同于先进先出队列的另一种队列.每次从队列中取出的是具有最高优先权的元素. PriorityQueue是从JDK1.5开始提供的新的数据结构接口. 如果不提供Comparator的话,优先 ...

  8. java PriorityBlockingQueue 基于优先级队列,的读出操作可以阻止.

    java PriorityBlockingQueue 基于优先级队列.的读出操作可以阻止. package org.rui.thread.newc; import java.util.ArrayLis ...

  9. 《转》JAVA中PriorityQueue优先级队列使用方法

    该文章转自:http://blog.csdn.net/hiphopmattshi/article/details/7334487 优先级队列是不同于先进先出队列的另一种队列.每次从队列中取出的是具有最 ...

随机推荐

  1. 20191010-6 alpha week 1/2 Scrum立会报告+燃尽图 04

    此作业的要求参见[https://edu.cnblogs.com/campus/nenu/2019fall/homework/8749] 一.小组情况 队名:扛把子 组长:迟俊文 组员:宋晓丽 梁梦瑶 ...

  2. JavaScript笔记三

    1.数据类型 - JS中一共分成六种数据类型 - String 字符串 - Number 数值 - Boolean 布尔值 - Null 空值 - Undefined 未定义 - Object 对象 ...

  3. selenium常用命令之操作页面元素及获取元素内容的事件整理

    /**id <input type="text" id="phone" name="phone" class="LoginT ...

  4. linux常见配置文件路径

    1:/etc/sysconfig/i18n                        (语言配置文件). 2:/etc/sysconfig/network-scripts/ifcfg-eth0   ...

  5. [从今天开始修炼数据结构]串、KMP模式匹配算法

    [从今天开始修炼数据结构]基本概念 [从今天开始修炼数据结构]线性表及其实现以及实现有Itertor的ArrayList和LinkedList [从今天开始修炼数据结构]栈.斐波那契数列.逆波兰四则运 ...

  6. c 语言I博客作业02

    这个作业属于哪个课程 C语言程序设计1 这个作业要求在哪里 https://edu.cnblogs.com/campus/zswxy/SE2019-2/homework/8687 我在这个课程的目标是 ...

  7. c语言博客专业05

    问题 答案 这个作业属于那个课程 C语言程序设计II 这个作业要求在哪里 https://edu.cnblogs.com/campus/zswxy/CST2019-2/homework/8655 我在 ...

  8. num += num 与 num = num+ num

    a = 100def test(num): num += num print(num) test(a)print(a) 200100 这里 num += num 与 num = num+ num 不能 ...

  9. 腾讯视频缓存 tdl 转 mp4

    找到腾讯视频->设置,看下缓存文件的目录地址,然后cmd,通过命令进行转化.  copy/b *.tdl 1.mp4

  10. 谷歌地图 API 开发之获取坐标以及街道详情

    自己的项目中有获取当前点击的坐标经纬度或者获取当前街道的信息的需求.估计这个对于新手来说,还是比较麻烦的,因为从官网上找这个也并不是很好找,要找好久的,运气好的可能会一下子找到. 献上自己写的测试案例 ...