习题33要求利用DelayQueue实现书中例子GreenhouseScheduler。后文程序我参考书中例子DelayQueueDemo完成的,运行结果如后文所示,与原GreenhouseScheduler基本一致,但是LightOn类似乎会少运行几次。
不知何种原因导致这种结果,还请javaeye的高手们指点!
----------------------------------------------------------------
package concurrency;
/*Exercise 33: (7) Modify GreenhouseScheduler.java so that it uses a DelayQueue instead of a ScheduledExecutor.*/
import static java.util.concurrent.TimeUnit.*;
import static net.mindview.util.Print.*;
import java.util.concurrent.*;
import java.util.*;
//定义用于DelayQueue的父类
class DTask implements Runnable, Delayed{
private int delay=0;
private long initialDelay, period;
private long trigger;
DTask(long initialDelay, long period){
this.initialDelay=initialDelay;
this.period=period;
trigger=System.nanoTime() + NANOSECONDS.convert(initialDelay, MILLISECONDS);
}
public long getDelay(TimeUnit unit) {
return unit.convert(
trigger - System.nanoTime(), NANOSECONDS);
}
public void setDelay(){
//此处设置Delay,使得DelayedTaskConsumer消费后延迟时间trigger增加period;
trigger=System.nanoTime() + NANOSECONDS.convert(period, MILLISECONDS);
}
public int compareTo(Delayed arg) {
DTask that = (DTask)arg;
if(trigger < that.trigger) return -1;
if(trigger > that.trigger) return 1;
return 0;
}
public void run() {try {
TimeUnit.MILLISECONDS.sleep(period);
} catch (InterruptedException e) {
print("sleep interrupted");
}}
}
//主类GreenhouseScheduler33,通过一系列DTask的子类实现
//开关灯、调温等操作
public class GreenhouseScheduler33 {
private volatile boolean light = false;
private volatile boolean water = false;
private String thermostat = "Day";
public synchronized String getThermostat() {
return thermostat;
}
public synchronized void setThermostat(String value) {
thermostat = value;
}
//LightOn
class LightOn extends DTask {
LightOn(long initialDelay, long period) {
super(initialDelay, period);
}
public void run() {
super.run();
// Put hardware control code here to
// physically turn on the light.
System.out.println("Turning on lights");
light = true;
}
}
class LightOff extends DTask {
LightOff(long initialDelay, long period) {
super(initialDelay, period);
}
public void run() {
// Put hardware control code here to
// physically turn off the light.
System.out.println("Turning off lights");
light = false;
}
}
class WaterOn extends DTask {
WaterOn(long initialDelay, long period) {
super(initialDelay, period);
}
public void run() {
// Put hardware control code here.
System.out.println("Turning greenhouse water on");
water = true;
}
}
class WaterOff extends DTask {
WaterOff(long initialDelay, long period) {
super(initialDelay, period);
}
public void run() {
// Put hardware control code here.
System.out.println("Turning greenhouse water off");
water = false;
}
}
class ThermostatNight extends DTask {
ThermostatNight(long initialDelay, long period) {
super(initialDelay, period);
}
public void run() {
// Put hardware control code here.
System.out.println("Thermostat to night setting");
setThermostat("Night");
}
}
class ThermostatDay extends DTask {
ThermostatDay(long initialDelay, long period) {
super(initialDelay, period);
}
public void run() {
// Put hardware control code here.
System.out.println("Thermostat to day setting");
setThermostat("Day");
}
}
class Bell extends DTask {
Bell(long initialDelay, long period) {
super(initialDelay, period);
}
public void run() { System.out.println("Bing!"); }
}
class Terminate extends DTask {
ExecutorService exec;
Terminate(long initialDelay, long period,ExecutorService exec) {
super(initialDelay, period);
this.exec=exec;
}
public void run() {
System.out.println("Terminating");
// Must start a separate task to do this job,
// since the scheduler has been shut down:
exec.shutdownNow();
new Thread() {
public void run() {
for(DataPoint d : data)
System.out.println(d);
}
}.start();
}
}
// New feature: data collection
static class DataPoint {
final Calendar time;
final float temperature;
final float humidity;
public DataPoint(Calendar d, float temp, float hum) {
time = d;
temperature = temp;
humidity = hum;
}
public String toString() {
return time.getTime() +
String.format(
" temperature: %1$.1f humidity: %2$.2f",
temperature, humidity);
}
}
private Calendar lastTime = Calendar.getInstance();
{ // Adjust date to the half hour
lastTime.set(Calendar.MINUTE, 30);
lastTime.set(Calendar.SECOND, 00);
}
private float lastTemp = 65.0f;
private int tempDirection = +1;
private float lastHumidity = 50.0f;
private int humidityDirection = +1;
private Random rand = new Random(47);
List<DataPoint> data = Collections.synchronizedList(
new ArrayList<DataPoint>());
class CollectData extends DTask {
CollectData(long initialDelay, long period) {
super(initialDelay, period);
}
public void run() {
System.out.println("Collecting data");
synchronized(GreenhouseScheduler33.this) {
// Pretend the interval is longer than it is:
lastTime.set(Calendar.MINUTE,
lastTime.get(Calendar.MINUTE) + 30);
// One in 5 chances of reversing the direction:
if(rand.nextInt(5) == 4)
tempDirection = -tempDirection;
// Store previous value:
lastTemp = lastTemp +
tempDirection * (1.0f + rand.nextFloat());
if(rand.nextInt(5) == 4)
humidityDirection = -humidityDirection;
lastHumidity = lastHumidity +
humidityDirection * rand.nextFloat();
// Calendar must be cloned, otherwise all
// DataPoints hold references to the same lastTime.
// For a basic object like Calendar, clone() is OK.
data.add(new DataPoint((Calendar)lastTime.clone(),
lastTemp, lastHumidity));
}
}
}
class DelayedTaskConsumer implements Runnable{
private DelayQueue<DTask> queue;
DelayedTaskConsumer(DelayQueue<DTask> queue){
this.queue=queue;
}
public void run(){
try {
while(!Thread.interrupted()){
DTask t=queue.take();
t.run();
//将取出对象延迟时间设置后重新放入队列
t.setDelay();
queue.put(t);
} // Run task with the current thread
} catch(InterruptedException e) {
// Acceptable way to exit
}
print("Finished DelayedTaskConsumer");
}
}
public static void main(String[] args) {
DelayQueue<DTask> queue=new DelayQueue<DTask>();
GreenhouseScheduler33 gh = new GreenhouseScheduler33();
ExecutorService exec=Executors.newCachedThreadPool();
queue.put(gh.new Bell(0, 1000));
queue.put(gh.new ThermostatNight(0, 2000));
queue.put(gh.new LightOn(0, 200));
queue.put(gh.new LightOff(0, 400));
queue.put(gh.new WaterOn(0, 600));
queue.put(gh.new WaterOff(0, 800));
queue.put(gh.new ThermostatDay(0, 1400));
queue.put(gh.new CollectData(500, 500));
//terminate
queue.put(gh.new Terminate(3000,3000,exec));
exec.execute(gh.new DelayedTaskConsumer(queue));
}
}
---------------------------------------------------------------
//output:
/*Bing!
Thermostat to night setting
Turning on lights
Turning off lights
Turning greenhouse water on
Turning greenhouse water off
Thermostat to day setting
Turning on lights
Turning off lights
Collecting data
Turning on lights
Turning greenhouse water on
Turning on lights
Turning greenhouse water off
Turning off lights
Bing!
Collecting data
Turning on lights
Turning greenhouse water on
Turning off lights
Turning on lights
Thermostat to day setting
Collecting data
Turning on lights
Turning greenhouse water off
Turning off lights
Turning greenhouse water on
Thermostat to night setting
Bing!
Turning on lights
Collecting data
Turning off lights
Turning on lights
Turning greenhouse water on
Turning on lights
Collecting data
Turning greenhouse water off
Turning off lights
Turning on lights
Thermostat to day setting
Terminating
Finished DelayedTaskConsumer
Sat Mar 05 15:00:00 GMT 2011 temperature: 66.4 humidity: 50.05
Sat Mar 05 15:30:00 GMT 2011 temperature: 68.0 humidity: 50.47
Sat Mar 05 16:00:00 GMT 2011 temperature: 69.7 humidity: 51.42
Sat Mar 05 16:30:00 GMT 2011 temperature: 70.8 humidity: 50.87
Sat Mar 05 17:00:00 GMT 2011 temperature: 72.0 humidity: 50.32
*/
分享到:
相关推荐
│ 高并发编程第二阶段44讲、被动引用和类加载过程的练习巩固训练题.mp4 │ 高并发编程第二阶段45讲、ClassLoader加载阶段发生的故事.mp4 │ 高并发编程第二阶段46讲、ClassLoader链接阶段(验证,准备,解析)...
在Java编程中,DelayQueue是一种特殊的并发队列,它遵循先进先出(FIFO)原则,但具有一个独特的特性:元素只有在其指定的延迟时间过去之后才能被获取和处理。这个特性使得DelayQueue成为实现定时任务和延迟操作的...
1. java.util.concurrent - Java 并发工具包 2. 阻塞队列 BlockingQueue 3. 数组阻塞队列 ArrayBlockingQueue 4. 延迟队列 DelayQueue 5. 链阻塞队列 LinkedBlockingQueue 6. 具有优先级的阻塞队列 ...
"Java多线程并发开发之DelayQueue使用示例" DelayQueue是Java多线程并发开发中的一种常用的数据结构,它是一个无界的BlockingQueue,用于放置实现了Delayed接口的对象。DelayQueue的主要作用是按照对象的延迟时间...
1. java.util.concurrent - Java 并发工具包 2. 阻塞队列 BlockingQueue 3. 数组阻塞队列 ArrayBlockingQueue 4. 延迟队列 DelayQueue 5. 链阻塞队列 LinkedBlockingQueue 6. 具有优先级的阻塞队列 ...
Java并发编程是现代编程中的一个重要组成部分,它涉及到多个线程同时执行,从而实现程序的高效率与多任务处理能力。在Java中,实现并发的方式主要有两种:一是使用线程,二是使用进程。进程是资源分配的基本单位,...
│ 高并发编程第二阶段44讲、被动引用和类加载过程的练习巩固训练题.mp4 │ 高并发编程第二阶段45讲、ClassLoader加载阶段发生的故事.mp4 │ 高并发编程第二阶段46讲、ClassLoader链接阶段(验证,准备,解析)...
【Java并发编程--BlockingQueue详解】 BlockingQueue 是 Java 并发包(java.util.concurrent)中的一个接口,它扩展了 Queue 接口,并引入了线程安全的特性,特别适合于多线程环境下的数据共享。 BlockingQueue 的...
java.util.concurrent - Java 并发工具包 2. 阻塞队列 BlockingQueue 3. 数组阻塞队列 ArrayBlockingQueue 4. 延迟队列 DelayQueue 5. 链阻塞队列 LinkedBlockingQueue 6. 具有优先级的阻塞队列 ...
通过以上对Java 8并发工具包中的阻塞队列、阻塞双端队列以及阻塞转移队列的详细介绍,我们可以看出Java 8在并发编程方面提供了强大的支持,使得开发者能够在处理高并发场景时更加得心应手。无论是通过阻塞队列管理...
在Java并发编程中,阻塞队列和阻塞栈是两个重要的并发数据结构,它们在多线程环境下的高效通信和资源管理中扮演着至关重要的角色。这些数据结构源自Java的并发包`java.util.concurrent`,是实现并发设计模式如生产者...
`DelayQueue`是Java并发库`java.util.concurrent`中的一个数据结构,它是一个基于优先级队列的无界阻塞队列,可以用于存储具有延迟时间的元素。而Redis则是一个高性能的键值数据库,通过其丰富的数据结构和操作,...
在Java的并发编程中,`DelayQueue`是一个非常特殊的队列,它属于并发包`java.util.concurrent`的一部分。`DelayQueue`是一个基于优先级队列(PriorityQueue)实现的无界阻塞队列,它的主要特性是元素只有在达到指定...
1. java.util.concurrent - Java 并发工具包 2. 阻塞队列 BlockingQueue 3. 数组阻塞队列 ArrayBlockingQueue 4. 延迟队列 DelayQueue 5. 链阻塞队列 LinkedBlockingQueue 6. 具有优先级的阻塞队列 ...
Java并发编程常见知识点源码集锦,涉及到对象锁,Executors多任务线程框架,线程池等示例,列出一些源码包中包括的内容: volatile关键字的非原子性、volatile关键字的使用、AtomicInteger原子性操作、线程安全小...
标题 "Delayed interface and Delay Queue" 涉及到Java并发编程中的两个重要概念:Delayed接口和DelayQueue。这篇博文可能是作者Tomboxfan在iteye博客上分享关于这两个概念的深入理解和应用实例。 Delayed接口是...
学习视频,可以丰富java知识。能够获得更多的专业技能
Java 利用 DelayedQueue 实现本地的延迟队列 DelayedQueue 是 Java 中的一种特殊的阻塞队列,它用于存放实现了 Delayed 接口的对象。这种队列是有序的,即队头对象的延迟到期时间最长。 DelayedQueue 能够满足我们...
Java并发容器是Java多线程编程中的重要工具,它们提供了高效、线程安全的数据结构,使得在并发环境下处理数据变得更加简单。在`java.util.concurrent`包中,有四种主要的并发容器类型:队列(BlockingQueue)、Map...