一、观察者模式定义
观察者模式(Observer Pattern),又叫做发布-订阅模式、模型-视图模式、源-监听器模式或从属者模式,它定义了对象之间的一对多依赖,这样一来,当一个对象改变状态时,它的所有依赖者都会收到通知并自动更新。
二、观察者模式优缺点
1、优点:
(1)观察者和被观察者之间建立了一种抽象的耦合关系,观察者依赖于被观察者,被观察者又依赖于观察者,这种关系更灵活。
(2)观察者模式可以实现表示层和数据逻辑层的分离,并支持松耦合,使得表示层可以独立于数据逻辑层变化。
2、缺点:
(1)如果一个被观察者对象有很多直接和间接的观察者的话,将所有的观察者都通知到会花费很多时间。
(2)如果在观察者和观察目标之间有循环依赖的话,观察目标会触发它们之间进行循环调用,可能导致系统崩溃。
三、观察者模式实现
观察者模式的实现,一般需要观察者和被观察者这两个角色。
观察者:它定义了一个更新接口,当被观察者调用notifyObservers()方法时,观察者的update()方法会被回调。
public interface Observer {
void update(String message);
}
被观察者:它也是一个接口,它提供了一些方法,使得增加,删除,通知观察者成为可能。
public interface Subject {
void registerObserver(Observer o);
void removeObserver(Observer o);
void notifyObservers();
}
实现被观察者接口:
public class WeatherData implements Subject {
private ArrayList<Observer> observers;
private float temperature;
private float humidity;
private float pressure;
public WeatherData() {
observers = new ArrayList<Observer>();
}
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 measurementsChanged() {
notifyObservers();
}
public void setMeasurements(float temperature, float humidity, float pressure) {
this.temperature = temperature;
this.humidity = humidity;
this.pressure = pressure;
measurementsChanged();
}
// other WeatherData methods here
}
实现观察者接口:
public class CurrentConditionsDisplay implements Observer {
private float temperature;
private float humidity;
private Subject weatherData;
public CurrentConditionsDisplay(Subject weatherData) {
this.weatherData = weatherData;
weatherData.registerObserver(this);
}
public void update(float temperature, float humidity, float pressure) {
this.temperature = temperature;
this.humidity = humidity;
display();
}
public void display() {
System.out.println("Current conditions: " + temperature + "F degrees and " + humidity + "% humidity");
}
}