OnJava8-Examples/threads/DelayQueueDemo.java

106 lines
3.2 KiB
Java
Raw Normal View History

2016-07-05 14:46:09 -06:00
// threads/DelayQueueDemo.java
2015-12-15 11:47:04 -08:00
// (c)2016 MindView LLC: see Copyright.txt
2015-11-15 15:51:35 -08:00
// We make no guarantees that this code is fit for any purpose.
2016-09-23 13:23:35 -06:00
// Visit http://OnJava8.com for more book information.
2015-06-15 17:47:35 -07:00
import java.util.concurrent.*;
import java.util.*;
import static java.util.concurrent.TimeUnit.*;
class DelayedTask implements Runnable, Delayed {
private static int counter = 0;
private final int id = counter++;
private final int delta;
private final long trigger;
protected static List<DelayedTask> sequence =
new ArrayList<>();
public DelayedTask(int delayInMilliseconds) {
delta = delayInMilliseconds;
trigger = System.nanoTime() +
NANOSECONDS.convert(delta, MILLISECONDS);
sequence.add(this);
}
@Override
public long getDelay(TimeUnit unit) {
return unit.convert(
trigger - System.nanoTime(), NANOSECONDS);
}
@Override
public int compareTo(Delayed arg) {
DelayedTask that = (DelayedTask)arg;
if(trigger < that.trigger) return -1;
if(trigger > that.trigger) return 1;
return 0;
}
@Override
2015-11-03 12:00:44 -08:00
public void run() { System.out.print(this + " "); }
2015-06-15 17:47:35 -07:00
@Override
public String toString() {
return String.format("[%1$-4d]", delta) +
" Task " + id;
}
public String summary() {
return "(" + id + ":" + delta + ")";
}
public static class EndSentinel extends DelayedTask {
private ExecutorService exec;
public EndSentinel(int delay, ExecutorService e) {
super(delay);
exec = e;
}
@Override
public void run() {
for(DelayedTask pt : sequence) {
2015-11-03 12:00:44 -08:00
System.out.print(pt.summary() + " ");
2015-06-15 17:47:35 -07:00
}
2015-11-03 12:00:44 -08:00
System.out.println();
System.out.println(this + " Calling shutdownNow()");
2015-06-15 17:47:35 -07:00
exec.shutdownNow();
}
}
}
class DelayedTaskConsumer implements Runnable {
private DelayQueue<DelayedTask> q;
public DelayedTaskConsumer(DelayQueue<DelayedTask> q) {
this.q = q;
}
@Override
public void run() {
try {
while(!Thread.interrupted())
2016-01-25 18:05:55 -08:00
q.take().run(); // Run task with current thread
2015-06-15 17:47:35 -07:00
} catch(InterruptedException e) {
// Acceptable way to exit
}
2015-11-03 12:00:44 -08:00
System.out.println("Finished DelayedTaskConsumer");
2015-06-15 17:47:35 -07:00
}
}
public class DelayQueueDemo {
public static void main(String[] args) {
2016-01-25 18:05:55 -08:00
SplittableRandom rand = new SplittableRandom(47);
ExecutorService es = Executors.newCachedThreadPool();
2015-06-15 17:47:35 -07:00
DelayQueue<DelayedTask> queue =
new DelayQueue<>();
// Fill with tasks that have random delays:
for(int i = 0; i < 20; i++)
queue.put(new DelayedTask(rand.nextInt(5000)));
// Set the stopping point
2016-01-25 18:05:55 -08:00
queue.add(new DelayedTask.EndSentinel(5000, es));
es.execute(new DelayedTaskConsumer(queue));
2015-06-15 17:47:35 -07:00
}
2015-09-07 11:44:36 -06:00
}
/* Output:
2016-07-22 14:45:35 -06:00
[70 ] Task 10 [125 ] Task 13 [267 ] Task 19 [635 ] Task 0
[650 ] Task 16 [682 ] Task 17 [807 ] Task 11 [1131] Task 18
[1177] Task 4 [1193] Task 9 [1634] Task 15 [1656] Task 6
[2400] Task 12 [3479] Task 5 [3737] Task 1 [3768] Task 7
[3941] Task 2 [4720] Task 3 [4762] Task 14 [4948] Task 8
(0:635) (1:3737) (2:3941) (3:4720) (4:1177) (5:3479)
(6:1656) (7:3768) (8:4948) (9:1193) (10:70) (11:807)
(12:2400) (13:125) (14:4762) (15:1634) (16:650) (17:682)
(18:1131) (19:267) (20:5000)
2015-06-15 17:47:35 -07:00
[5000] Task 20 Calling shutdownNow()
Finished DelayedTaskConsumer
2015-09-07 11:44:36 -06:00
*/