本篇文章主要用来介绍DeleyQueue的使用方式,根据这个类的名字就可以判断,这是一个Queue,且从它里面去的元素的顺序是按照时间的延迟来的.网上有很多例子,可是都不是很详细,我把书上的例子看了一下,然后用自己的思路写出来,原来发现里面还是有很多的误区,下面的程序是例子,加了一些注释:
/** * */ package com.eric.concurrency; import java.util.ArrayList; import java.util.List; import java.util.Random; import java.util.concurrent.DelayQueue; import java.util.concurrent.Delayed; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; /** * this class used to explain how to use delayqueue * @author Eric * */ public class DelayQueueDemo { public static void main(String[] args) { try { ExecutorService es = Executors.newCachedThreadPool(); DelayQueue<DelayTask> queue = new DelayQueue<DelayTask>(); Random rand = new Random(); // add 20's DelayTask to DelayQueue for (int i = 0; i < 20; i++) { queue.add(new DelayTask(rand.nextInt(500))); } EndTask endTaks = new EndTask(501); // add endtask to DelayQueue, it's time is 50001 to ensure will be // execute at the end of queue queue.add(endTaks); es.execute(new DelayConsumer(queue)); TimeUnit.SECONDS.sleep(1); es.shutdownNow(); } catch (InterruptedException e) { e.printStackTrace(); } } } class DelayConsumer implements Runnable { private DelayQueue<DelayTask> queue; public DelayConsumer(DelayQueue<DelayTask> queue) { this.queue = queue; } public void run() { try { System.out.println("############print order by DelayQueue###############"); // from DelayQueue get DelayTask and execute it while (!Thread.interrupted()) { queue.take().run(); } } catch (InterruptedException e) { System.out.println("Exit Consumer"); } } } class EndTask extends DelayTask { public EndTask(long delay) { super(delay); } @Override public void run() { try { TimeUnit.MILLISECONDS.sleep(1); // print all task from internal list System.out.println("#############print by list order###############"); for (DelayTask delayTask : tasks) { System.out.println("end" + delayTask); } } catch (InterruptedException e) { System.out.println("Exit LIST"); } } } class DelayTask implements Runnable, Delayed { protected static List<DelayTask> tasks = new ArrayList<DelayTask>(); public long time = 0; public long delay; private static int count = 0; private int id = count++; public DelayTask(long delay) { this.delay = delay; time = System.nanoTime() + TimeUnit.NANOSECONDS.convert(this.delay, TimeUnit.MICROSECONDS); tasks.add(this); } @Override public int compareTo(Delayed o) { long result = ((DelayTask) o).getTime() - this.getTime(); if (result > 0) { return 1; } if (result < 0) { return -1; } return 0; } @Override public long getDelay(TimeUnit unit) { return unit.convert(time - System.nanoTime(), TimeUnit.NANOSECONDS); } @Override public void run() { System.out.println(this); } public String toString() { // use time to print for get real "delay time" return "Task " + id + ":" + time; } public long getTime() { return time; } public void setTime(long time) { this.time = time; } }这个程序运行的结果分两部分,一部分是从QUEUE中取出来的元素,另一部分是从普通的链表里拿出来的展示的结果.所有的元素按照相同的顺序分别放到queue和list中,最后展现的顺序是不同的,在list中的元素是按照放入的顺序取出的,而DelayQueue中的元素是按照Delay的delay因素来决定取出的顺序