1.什么是队列

队列是一种特殊的线性表,它只允许在表的前端(front)进行删除操作,而在表的后端(rear)进行插入操作。进行插入操作的端称为队尾,进行删除操作的端称为队头。队列中没有元素时,称为空队列。

    在队列这种数据结构中,最先插入的元素将是最先被删除的元素;反之最后插入的元素将是最后被删除的元素,因此队列又称为“先进先出”(FIFO—first in first out)的线性表。


2.队列常用的方法


(1).add:增加一个元索,如果队列已满,则抛出一个IIIegaISlabEepeplian异常;

(2).remove:移除并返回队列头部的元素 ,如果队列为空,则抛出一个NoSuchElementException异常;

(3).element:返回队列头部的元素,如果队列为空,则抛出一个NoSuchElementException异常;

(4).offer:添加一个元素并返回true,如果队列已满,则返回false;

(5).poll:移除并返问队列头部的元素,如果队列为空,则返回null;

(6).peek:返回队列头部的元素, 如果队列为空,则返回null;        

(7).put:添加一个元素,如果队列满,则阻塞;

(8).take:移除并返回队列头部的元素,如果队列为空,则阻塞;


3.队列(Queue)的类关系图

java 队列 取出全部数据 java队列删除元素_数据结构


4.PriorityQueue(优先队列)


优先级队列是不同于先进先出队列的另一种队列。每次从队列中取出的是具有最高优先权的元素。如果不提供Comparator的话,优先队列中元素默认按自然顺序排列,也就是数字默认是小的在队列头,字符串则按字典序排列。


5.BlockingQueue(阻塞队列)

    BlockingQueue,顾名思义,阻塞队列。BlockingQueue是在java.util.concurrent下的,因此不难理解,BlockingQueue是为了解决多线程中数据高效安全传输而提出的。

    多线程中,很多场景都可以使用队列实现,比如经典的生产者/消费者模型,通过队列可以便利地实现两者之间数据的共享,定义一个生产者线程,定义一个消费者线程,通过队列共享数据就可以了。

    当然现实不可能都是理想的,比如消费者消费速度比生产者生产的速度要快,那么消费者消费到 一定程度上的时候,必须要暂停等待一下了(使消费者线程处于WAITING状态)。BlockingQueue的提出,就是为了解决这个问题的,他不用程序员去控制这些细节,同时还要兼顾效率和线程安全。

    阻塞队列所谓的"阻塞",指的是某些情况下线程会挂起(即阻塞),一旦条件满足,被挂起的线程又会自动唤醒。使用BlockingQueue,不需要关心什么时候需要阻塞线程,什么时候需要唤醒线程,这些内容BlockingQueue都已经做好了。


6.常见的BlockingQueue介绍

  (1).ArrayBlockingQueue: 基于数组的阻塞队列,必须指定队列大小。 比较简单。ArrayBlockingQueue中只有一个ReentrantLock对象,这意味着生产者和消费者无法并行运行(见下面的代码)。另 外,创建ArrayBlockingQueue时,可以指定ReentrantLock是否为公平锁,默认采用非公平锁。

  (2).LinkedBlockingQueue: 基于链表的阻塞队列,和ArrayBlockingQueue差不多。不过LinkedBlockingQueue如果不指定队列容量大小,会默认 一个类似无限大小的容量,之所以说是类似是因为这个无限大小是Integer.MAX_VALUE,这么说就好理解ArrayBlockingQueue 为什么必须要制定大小了,如果ArrayBlockingQueue不指定大小的话就用Integer.MAX_VALUE,那将造成大量的空间浪费,但是基于链表实现就不一样的,一个一个节点连起来而已。另外,LinkedBlockingQueue生产者和消费者都有自己的锁(见下面的代码),这意味着生产者和消费者可以"同时"运行。

  (3).PriorityBlockingQueue: PriorityBlockingQueue是一个没有边界的队列,它的排序规则和 java.util.PriorityQueue一样。需要注意,PriorityBlockingQueue中允许插入null对象。

  (4).SynchronousQueue: 一种没有缓冲的等待队列。每一个插入操作必须等待一个线程对应的移除操作。SynchronousQueue又有两种模式:公平模式和非公平模式。

  (5).DelayQueue:延迟型阻塞队列

  DelayQueue是一个BlockingQueue,其特化的参数是Delayed。Delayed扩展了Comparable接口,比较的基准为延时的时间值,Delayed接口的实现类getDelay的返回值应为固定值(final)。DelayQueue内部是使用PriorityQueue实现的。
  简单来说:
DelayQueue = BlockingQueue + PriorityQueue + Delayed
DelayQueue是一个使用优先队列(PriorityQueue)实现的BlockingQueue,优先队列的比较基准值是时间。




7.DelayQueue适合使用的一些实际开发场景

  (1).关闭空闲连接。服务器中,有很多客户端的连接,空闲一段时间之后需要关闭之。
  (2).缓存。缓存中的对象,超过了空闲时间,需要从缓存中移出。
  (3).任务超时处理。在网络协议滑动窗口请求应答式交互时,处理超时未响应的请求。




8.PriorityQueue代码示例


package com.biyao.current.priorityqueue;
/**
 * @ClassName: Student
 * @Description: 学生类实体
 * @author yangy
 * @date 2017年11月14日 下午3:04:41
 */
public class Student {
	
	private int id;
	
	private String name;
	
	public Student(int id,String name){
		this.id = id;
		this.name = name;
	}
	public int getId() {
		return id;
	}
	public void setId(int id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	
}
package com.biyao.current.priorityqueue;

import java.util.Comparator;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.Random;

/**
 * @ClassName: PriorityQueueExample
 * @Description: 优先队列测试类
 * @author yangy
 * @date 2017年11月14日 下午3:13:20
 */
public class PriorityQueueExample {
	
	public static void main(String[] args) {
		
		//优先队列自然排序示例
        Queue<Integer> integerPriorityQueue = new PriorityQueue<Integer>(7);
        Random rand = new Random();
        for(int i=0;i<7;i++){
            integerPriorityQueue.add(new Integer(rand.nextInt(100)));
        }
        for(int i=0;i<7;i++){
            Integer in = integerPriorityQueue.poll();
            System.out.println("Processing Integer:"+in);
        }
        
      //Comparator实现
       Comparator<Student> idComparator = new Comparator<Student>(){
	        @Override
	        public int compare(Student c1, Student c2) {
	            return (int) (c1.getId() - c2.getId());
	        }
        };
        
        Queue<Student> studentPriorityQueue = new PriorityQueue<Student>(7,idComparator);
        for(int i=0; i<7; i++){
            int id = rand.nextInt(100);
            studentPriorityQueue.add(new Student(id, "yangy "+id + "号"));
        }
        
        while(true){
        	Student student = studentPriorityQueue.poll();
            if(student == null){
            	break;
            }
            System.out.println("学生信息:【id="+student.getId()+",name="+student.getName()+"】");
        }
		
	}
	
}


运行效果图:

java 队列 取出全部数据 java队列删除元素_java 队列 取出全部数据_02


9.DelayQueue代码示例


package com.biyao.current.delayqueue2;

import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit;

/**
 * @ClassName: Student
 * @Description: 学生实体类
 * @author yangy
 * @date 2017年11月17日 下午1:10:57
 */
public class Student implements Runnable,Delayed{
	
	  private String name;  //姓名
	  private long costTime;//做试题的时间
	  private long finishedTime;//完成时间

	  public Student(String name, long costTime) {
	         this. name = name;
	         this. costTime= costTime;
	         finishedTime = costTime + System. currentTimeMillis();
	  }

	  @Override
	  public void run() {
	        System. out.println( name + " 交卷,用时" + costTime /1000 + "s");
	  }

	  @Override
	  public long getDelay(TimeUnit unit) {
	         return ( finishedTime - System.currentTimeMillis());
	  }

	  @Override
	  public int compareTo(Delayed o) {
	        Student other = (Student) o;
	         return costTime >= other. costTime?1:-1;
	  }
}


package com.biyao.current.delayqueue2;

import java.util.Random;
import java.util.concurrent.DelayQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * @ClassName: Teacher
 * @Description: 老师实体类
 * @author yangy
 * @date 2017年11月17日 下午1:12:48
 */
public class Teacher {
	
	  static final int STUDENT_SIZE = 30;
	  public static void main(String[] args) throws InterruptedException {
	        Random r = new Random();
	        //把所有学生看做一个延迟队列
	        DelayQueue<Student> students = new DelayQueue<Student>();
	        
	        students.put( new Student( "学生-yangy", 10000));
	        Student student = students.take();
	        student.run();
	        
	        //构造一个线程池用来让学生们“做作业”
	        ExecutorService exec = Executors.newFixedThreadPool(STUDENT_SIZE);
	         for ( int i = 0; i < STUDENT_SIZE; i++) {
               //初始化学生的姓名和做题时间
               students.put( new Student( "学生" + (i + 1), 3000 + r.nextInt(10000)));
	        }
	        //开始做题
	        while(! students.isEmpty()){
	           exec.execute( students.take());
	        }
	        exec.shutdown();
	        
	  }
}


运行效果图:


java 队列 取出全部数据 java队列删除元素_java_03