接上篇。
观察者模式实现的气象站
WeatherData类作为主题,需要实现观察者的注册、删除以及通知方法。
public class WeatherData {
// 实例变量声明
public void measurementsChanged() {
// 存放观察者
private ArrayList observers;
private float temp = getTemperature();
private float humidity = getHumidity();
private float pressure = getPressure();
public WeatherData() {
observers = new ArrayList();
}
public void registerObserver(Observer o) {
observers.add(o);
}
public void removeObserver(Observer o) {
int i = observers.indexOf(o);
if(i > 0) {
observers.remove(i);
}
}
public void notifyObservers() {
for(int i = 0; i < observers.size(); i++) {
Observer observer = (Observer)observers.get(i);
observer.update(temperature, humidity, pressure);
}
}
public void meansurementsChanged() {
notifyObservers();
}
}
}
现在已经把WeatherData类写出来了,现在轮到布告板了。先来看看目前状况布告板,其他两个布告板类似。
public class CurrentConditionDisplay implements Observer {
private float temperature;
private float humidity;
private float pressure;
private Subject weatherData;
public CurrentConditionDisplay(Subject weatherData) {
this.weatherData = weatherData;
weatherData.registerObserver(this);
}
public void update(float temperature, float humidity, float pressure) {
this.temperature = temperature;
this.humidity = humidity;
this.pressure = pressure;
display();
}
public void display() {
System.out.println("Current conditions: " + temperature + " and " + humidity);
}
}
布告板类需要保存对主题的引用,因为在以后取消注册,已经有了对主题的引用会比较方便。
另外,可以增加setChanged()方法来标记改变状态,好让notifyObservers()知道当它被调用时应该更新观察者。如果调用notifyObservers()之前没有先调用setChanged(),观察者就不会被通知。
这样做有其必要性。setChanged()方法可以让你在更新观察者时,有更多的弹性,你可以更适当地通知观察者。比如,如果没有setChanged()方法,气象站测量是如此敏锐,以至于温度计每十分之一度就会更新,这会造成WeatherData对象持续不断地通知观察者,我们并不希望看到这样的事情发生。如果我们希望半度以上才更新,就可以在温度差距到达半度时,调用setChanged(),进行有效的更新。
小结
观察者定义了对象之间一对多的关系
主题,也就是可观察者用一个共同的接口来更新观察者。
观察者和可观察者之间用松耦合方式结合(loosecoupling),可观察者不知道观察者的细节,只知道观察者实现了观察者接口。
使用此模式时,你可从被观察者处推(push)或拉(pull)数据,然而,推的方式被认为更正确。
有对各观察者时,不可以依赖特定的通知次序。