OnJava8-Examples/threads/GreenhouseScheduler.java

203 lines
6.1 KiB
Java
Raw Normal View History

2016-07-05 14:46:09 -06:00
// threads/GreenhouseScheduler.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
// Rewriting innerclasses/GreenhouseController.java
2016-01-25 18:05:55 -08:00
// to use a ScheduledThreadPoolExecutor
2016-07-28 13:42:03 -06:00
// {java GreenhouseScheduler 5000}
2015-06-15 17:47:35 -07:00
import java.util.concurrent.*;
import java.util.*;
public class GreenhouseScheduler {
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;
}
ScheduledThreadPoolExecutor scheduler =
new ScheduledThreadPoolExecutor(10);
public void schedule(Runnable event, long delay) {
scheduler.schedule(event,delay,TimeUnit.MILLISECONDS);
}
public void
repeat(Runnable event, long initialDelay, long period) {
scheduler.scheduleAtFixedRate(
event, initialDelay, period, TimeUnit.MILLISECONDS);
}
class LightOn implements Runnable {
@Override
public void run() {
// Put hardware control code here to
// physically turn on the light.
System.out.println("Turning on lights");
light = true;
}
}
class LightOff implements Runnable {
@Override
public void run() {
// Put hardware control code here to
// physically turn off the light.
System.out.println("Turning off lights");
light = false;
}
}
class WaterOn implements Runnable {
@Override
public void run() {
// Put hardware control code here.
System.out.println("Turning greenhouse water on");
water = true;
}
}
class WaterOff implements Runnable {
@Override
public void run() {
// Put hardware control code here.
System.out.println("Turning greenhouse water off");
water = false;
}
}
class ThermostatNight implements Runnable {
@Override
public void run() {
// Put hardware control code here.
System.out.println("Thermostat to night setting");
setThermostat("Night");
}
}
class ThermostatDay implements Runnable {
@Override
public void run() {
// Put hardware control code here.
System.out.println("Thermostat to day setting");
setThermostat("Day");
}
}
class Bell implements Runnable {
@Override
public void run() { System.out.println("Bing!"); }
}
class Terminate implements Runnable {
@Override
public void run() {
System.out.println("Terminating");
scheduler.shutdownNow();
// Must start a separate task to do this job,
// since the scheduler was shut down:
new Thread() {
@Override
public void run() {
for(DataPoint d : data)
System.out.println(d);
}
}.start();
}
}
// New feature: data collection
static class DataPoint {
final Calendar time;
2016-01-25 18:05:55 -08:00
final double temperature;
final double humidity;
public DataPoint(Calendar d, double temp, double hum) {
2015-06-15 17:47:35 -07:00
time = d;
temperature = temp;
humidity = hum;
}
@Override
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);
}
2016-01-25 18:05:55 -08:00
private double lastTemp = 65.0f;
2015-06-15 17:47:35 -07:00
private int tempDirection = +1;
2016-01-25 18:05:55 -08:00
private double lastHumidity = 50.0f;
2015-06-15 17:47:35 -07:00
private int humidityDirection = +1;
2016-01-25 18:05:55 -08:00
private SplittableRandom rand = new SplittableRandom(47);
2015-06-15 17:47:35 -07:00
List<DataPoint> data = Collections.synchronizedList(
new ArrayList<>());
class CollectData implements Runnable {
@Override
public void run() {
System.out.println("Collecting data");
synchronized(GreenhouseScheduler.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:
2016-01-25 18:05:55 -08:00
lastTemp +=
tempDirection * (1.0f + rand.nextDouble());
2015-06-15 17:47:35 -07:00
if(rand.nextInt(5) == 4)
humidityDirection = -humidityDirection;
2016-01-25 18:05:55 -08:00
lastHumidity +=
humidityDirection * rand.nextDouble();
2015-06-15 17:47:35 -07:00
// Calendar must be cloned, otherwise all
2016-01-25 18:05:55 -08:00
// DataPoints hold references to same lastTime.
// For basic object like Calendar, clone() is OK.
2015-06-15 17:47:35 -07:00
data.add(new DataPoint((Calendar)lastTime.clone(),
lastTemp, lastHumidity));
}
}
}
public static void main(String[] args) {
GreenhouseScheduler gh = new GreenhouseScheduler();
gh.schedule(gh.new Terminate(), 5000);
// Former "Restart" class not necessary:
gh.repeat(gh.new Bell(), 0, 1000);
gh.repeat(gh.new ThermostatNight(), 0, 2000);
gh.repeat(gh.new LightOn(), 0, 200);
gh.repeat(gh.new LightOff(), 0, 400);
gh.repeat(gh.new WaterOn(), 0, 600);
gh.repeat(gh.new WaterOff(), 0, 800);
gh.repeat(gh.new ThermostatDay(), 0, 1400);
gh.repeat(gh.new CollectData(), 500, 500);
}
2015-09-07 11:44:36 -06:00
}
2016-07-20 06:32:39 -06:00
/* Output: (First and Last 10 Lines)
2015-06-15 17:47:35 -07:00
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 on lights
2016-07-22 14:45:35 -06:00
Turning off lights
...________...________...________...________...
Terminating
2016-07-27 11:12:11 -06:00
Wed Jul 27 11:00:00 MDT 2016 temperature: 66.3 humidity:
2016-07-22 14:45:35 -06:00
50.20
2016-07-27 11:12:11 -06:00
Wed Jul 27 11:30:00 MDT 2016 temperature: 67.3 humidity:
2016-07-22 14:45:35 -06:00
51.02
2016-07-27 11:12:11 -06:00
Wed Jul 27 12:00:00 MDT 2016 temperature: 68.8 humidity:
2016-07-22 14:45:35 -06:00
51.82
2016-07-27 11:12:11 -06:00
Wed Jul 27 12:30:00 MDT 2016 temperature: 70.0 humidity:
2016-07-22 14:45:35 -06:00
52.19
2016-07-27 11:12:11 -06:00
Wed Jul 27 13:00:00 MDT 2016 temperature: 71.5 humidity:
2016-07-22 14:45:35 -06:00
52.99
2016-07-27 11:12:11 -06:00
Wed Jul 27 13:30:00 MDT 2016 temperature: 73.0 humidity:
2016-07-22 14:45:35 -06:00
53.32
2016-07-27 11:12:11 -06:00
Wed Jul 27 14:00:00 MDT 2016 temperature: 74.4 humidity:
2016-07-22 14:45:35 -06:00
53.99
2016-07-27 11:12:11 -06:00
Wed Jul 27 14:30:00 MDT 2016 temperature: 75.5 humidity:
2016-07-22 14:45:35 -06:00
54.40
2016-07-27 11:12:11 -06:00
Wed Jul 27 15:00:00 MDT 2016 temperature: 76.8 humidity:
2016-07-22 14:45:35 -06:00
53.96
2015-09-07 11:44:36 -06:00
*/