在上一篇中我用一张图来梳理了一下Java中的各种Queue之间的关系。这里介绍下PriorityQueue。PriorityQueue位于Java util包中,观其名字前半部分的单词Priority是优先的意思,实际上这个队列就是具有“优先级”。既然具有优先级的特性,那么就得有个前后排序的“规则”。所以其接受的类需要实现Comparable 接口。

API

1.构造函数

  1. PriorityQueue()
  2. PriorityQueue(Collection<? extends E> c)
  3. PriorityQueue(int initialCapacity)
  4. PriorityQueue(int initialCapacity, Comparator<? super E> comparator)
  5. PriorityQueue(PriorityQueue<? extends E> c)
  6. PriorityQueue(SortedSet<? extends E> c)

2.常用功能函数

方法名 功能描述
add(E e) 添加元素
clear() 清空
contains(Object o) 检查是否包含当前参数元素
offer(E e) 添加元素
peek() 读取元素,(不删除)
poll() 取出元素,(删除)
remove(Object o) 删除指定元素
size() 返回长度

用法示例

上面提到具有优先级,那么这里举个例子。我在上高中的时候,每月分一次班级,老师会按照本月月考的成绩来让每位同学优先选择自己心仪的座位。这里所有的同学便是一个队列;每次喊一个人进来挑选座位,这便是出对的操作;成绩由前至后,这边是优先的策略。

  • 代码示例如下:
  1. public class PriorityQueueTest {
  2. public static void main(String[] args) {
  3. final PriorityQueue<Student> queue=new PriorityQueue<>();
  4. Student p1=new Student(95,"张三");
  5. Student p2=new Student(89,"李四");
  6. Student p3=new Student(89,"李四");
  7. Student p4=new Student(67,"王五");
  8. Student p5=new Student(92,"赵六");
  9. queue.add(p1);
  10. queue.add(p2);
  11. queue.add(p3);//add 和offer效果一样。
  12. queue.offer(p4);//add 方法实现,其实就是调用了offer
  13. queue.offer(p5)
  14. for (Student Student : queue) {
  15. System.out.println(Student.toString());
  16. }
  17. System.out.println("---------------------");
  18. while(!queue.isEmpty()){
  19. System.out.println(queue.poll());
  20. }
  21. }
  22. }
  23. class Student implements Comparable{
  24. private int score;
  25. private String name;
  26. public Student(int age,String name){
  27. this.score=age;
  28. this.name=name;
  29. }
  30. public int getScore() {
  31. return score;
  32. }
  33. public void setScore(int score) {
  34. this.score = score;
  35. }
  36. public String getName() {
  37. return name;
  38. }
  39. public void setName(String name) {
  40. this.name = name;
  41. }
  42. public String toString(){
  43. return "姓名:"+name+"-"+score+"分";
  44. }
  45. @Override
  46. public int compareTo(Object o) {
  47. Student current=(Student)o;
  48. if(current.getScore()>this.score){
  49. return 1;
  50. }else if(current.getScore()==this.score){
  51. return 0;
  52. }
  53. return -1;
  54. }
  55. }
  • 运行结果:

    姓名:张三-95分

    姓名:赵六-92分

    姓名:王五-67分

    姓名:李四-89分

    ---------按顺序出队选座位------------

    姓名:张三-95分

    姓名:赵六-92分

    姓名:李四-89分

    姓名:李四-89分

    姓名:王五-67分

从第一部分输出可以看出,学生入队并不是 按顺序的,而在poll出来的时候是按顺序出队的,这里确实实现了分数高这优先选座位的效果,poll方法返回的总是队列剩余学生中分数最高的。

安全性

查看PriorityQueue类的源码,会发现增加操作,并不是原子操作。没有使用任何锁。那么,如果是在多线程环境,肯定是不安全的。下面给出例子,开启多个线程,调用同一个方法对Queue进行添加元素。然后输出结果。

  1. public class PriorityQueueTest {
  2. static final PriorityQueue<Integer> queue=new PriorityQueue<>();
  3. /**
  4. * 向队列中插入元素
  5. * @param number
  6. */
  7. public void add(int number){
  8. if(!queue.contains(number)){
  9. System.out.println(Thread.currentThread()+":"+number);
  10. queue.add(number);
  11. }
  12. }
  13. public static void main(String[] args) throws InterruptedException {
  14. final PriorityQueueTest qt=new PriorityQueueTest();
  15. final Random r=new Random();
  16. Thread t1=new Thread(){
  17. public void run(){
  18. System.out.println("t1开始运行...");
  19. for(int i=0;i<10;i++){
  20. qt.add(r.nextInt(10));
  21. }
  22. }
  23. };
  24. Thread t2=new Thread(){
  25. public void run(){
  26. System.out.println("t2开始运行...");
  27. for(int i=0;i<10;i++){
  28. qt.add(r.nextInt(10));
  29. }
  30. }
  31. };
  32. Thread t3=new Thread(){
  33. public void run(){
  34. System.out.println("t3开始运行...");
  35. for(int i=0;i<10;i++){
  36. qt.add(r.nextInt(10));
  37. }
  38. }
  39. };
  40. t1.start();
  41. t2.start();
  42. t3.start();
  43. t1.join();
  44. t2.join();
  45. t3.join();
  46. System.out.println("------ 运行结果 ---------");
  47. while(!queue.isEmpty()){
  48. System.out.println(queue.poll());
  49. }
  50. }
  51. }
  • 运行结果

    t2开始运行...

    t3开始运行...

    t1开始运行...

    ------ 运行结果 ---------

    0

    1

    1

    2

    3

    4

    5

    6

    7

    8

    9

    9

结果中我们可以看到,具有两个1,两个9.这是不符合我们预期的,我们预期的是not contains 才插入,现在的出现的了重复的。上面的例子只需要在add方法上加锁,才可以达到我们预期的效果。所以说,PriorityQueue非线程安全的。

[1]: http://www.cnblogs.com/demingblog/p/6474865.html

工作6年,失业19天

Java8系列- 如何用Java8 Stream API找到心仪的女朋友

Java8系列- 何用Java8 Stream API进行数据抽取与收集

spring如何启动的?这里结合spring源码描述了启动过程

SpringMVC是怎么工作的,SpringMVC的工作原理

spring 异常处理。结合spring源码分析400异常处理流程及解决方法

Mybatis Mapper接口是如何找到实现类的-源码分析

使用Netty实现HTTP服务器

Netty实现心跳机制

Netty系列

# Java Queue系列之PriorityQueue的更多相关文章

  1. Java 集合系列之四:Queue基本操作

    1. Java Queue 1. Java Queue 重要观点 Java Queue接口是Java Collections Framework的成员. Queue 实现通常不允许插入 null 元素 ...

  2. Java多线程系列十——BlockingQueue

    参考资料:http://ifeve.com/java-synchronousqueue/http://www.cnblogs.com/jackyuj/archive/2010/11/24/188655 ...

  3. Java多线程系列--“JUC锁”03之 公平锁(一)

    概要 本章对“公平锁”的获取锁机制进行介绍(本文的公平锁指的是互斥锁的公平锁),内容包括:基本概念ReentrantLock数据结构参考代码获取公平锁(基于JDK1.7.0_40)一. tryAcqu ...

  4. Java多线程系列--“JUC锁”04之 公平锁(二)

    概要 前面一章,我们学习了“公平锁”获取锁的详细流程:这里,我们再来看看“公平锁”释放锁的过程.内容包括:参考代码释放公平锁(基于JDK1.7.0_40) “公平锁”的获取过程请参考“Java多线程系 ...

  5. Java多线程系列--“JUC锁”09之 CountDownLatch原理和示例

    概要 前面对"独占锁"和"共享锁"有了个大致的了解:本章,我们对CountDownLatch进行学习.和ReadWriteLock.ReadLock一样,Cou ...

  6. Java多线程系列--“JUC锁”05之 非公平锁

    概要 前面两章分析了"公平锁的获取和释放机制",这一章开始对“非公平锁”的获取锁/释放锁的过程进行分析.内容包括:参考代码获取非公平锁(基于JDK1.7.0_40)释放非公平锁(基 ...

  7. Java多线程系列--“JUC锁”07之 LockSupport

    概述 本章介绍JUC(java.util.concurrent)包中的LockSupport.内容包括:LockSupport介绍LockSupport函数列表LockSupport参考代码(基于JD ...

  8. Java多线程系列--“JUC锁”08之 共享锁和ReentrantReadWriteLock

    概要 Java的JUC(java.util.concurrent)包中的锁包括"独占锁"和"共享锁".在“Java多线程系列--“JUC锁”02之 互斥锁Ree ...

  9. Java多线程系列--“JUC锁”11之 Semaphore信号量的原理和示例

    概要 本章,我们对JUC包中的信号量Semaphore进行学习.内容包括:Semaphore简介Semaphore数据结构Semaphore源码分析(基于JDK1.7.0_40)Semaphore示例 ...

随机推荐

  1. springboot +thymeleaf+myql 记录

    thymeleaf官方文档: https://www.thymeleaf.org/doc/tutorials/3.0/usingthymeleaf.pdf demo案例:https://github. ...

  2. 解决mysql中文乱码问题 在url后面添加?characterEncoding=utf8

  3. BZOJ4241历史研究——回滚莫队

    题目描述 IOI国历史研究的第一人——JOI教授,最近获得了一份被认为是古代IOI国的住民写下的日记.JOI教授为了通过这份日记来研究古代IOI国的生活,开始着手调查日记中记载的事件. 日记中记录了连 ...

  4. 洛谷P3953逛公园

    题目 作为\(NOIp2017D1T3\) 这个题还是很良心的,至少相对于\(NOIp2018\)来说,希望\(NOIp2019\)不会这么坑吧. 这个题可以作为记忆化搜索的进阶题了,做这个题的方法也 ...

  5. 天气预测(CNN)

    import torch import torch.nn as nn import torch.utils.data as Data import numpy as np import pymysql ...

  6. python学习day10 函数Ⅱ(参数&作用域)

    函数Ⅱ(参数&作用域) 知识小结: py2与py3的区别 逻辑运算()>not>and>or 字符串翻转切片翻转 is与==区别 git相关 数据类型判断 操作系统:cent ...

  7. pytest 11 allure2生成html报告

    allure是一个report框架,支持java的Junit/testng等框架,当然也可以支持python的pytest框架,也可以集成到Jenkins上展示高大上的报告界面. 环境准备 1.pyt ...

  8. redis-sentinel高可用配置(2)

    一:说明 前面我们已经配置了redis的主从配置(链接),这种主从架构有一个问题,当主master出现了故障了,怎么切换到从服务器上呢? 第一种:手动切换, 这种肯定会造成比较长一段时间的用户不能访问 ...

  9. Docker下安装GitLab

    1.需要先安装Docker和Docker Compose,参考:https://www.cnblogs.com/hackyo/p/9280042.html 2.配置GitLab SSL(可跳过): m ...

  10. 第三节:框架前期准备篇之利用Newtonsoft.Json改造MVC默认的JsonResult

    一. 背景 在MVC框架中,我们可能经常会用到 return Json(),而Json方法内部又是一个JsonResult类,那么JsonResult内部又是什么原理呢?在MVC框架中,各种xxxRe ...