基本原理
- 观察者模式类似于订牛奶业务:奶站是Subject,用户是Observer
- Subject:登记注册、删除和通知。registerObserver注册;removeObserver移除;modifyObservers()通知所有的注册的用户,根据不同的需求,可以是更新数据,让用户来取,也可能是实时推送,看具体需求定
- Observer:接收输入
- 观察者模式:对象之间多对一依赖的一种设计方案,被依赖的对象为Subject,依赖的对象为Observer,Subject通知Observer变化,比如这里的奶站是Subject,是1的一方。用户是Observer,是多的一方。
案例:天气预报
- 需求
- 气象站可以将每天测量到的温度,湿度,气压等等以公告的形式发布出去(比如发布到自己的网站或第三方)。
- 需要设计开放型API,便于其他第三方也能接入气象站获取数据。
- 提供温度、气压和湿度的接口
- 测量数据更新时,要能实时的通知给第三方
- UML类图
- 主要代码
1)Subject:
public interface Subject {
public void registerObserver(Observer o);
public void removeObserver(Observer o);
public void notifyObservers();
}
2)Subject实现类WeatherData:
import java.util.ArrayList;
import java.util.List;
/**
* 核心类
* 天气信息
* 管理观察者
*/
public class WeatherData implements Subject {
private float temperature;
private float pressure;
private float humidity;
//观察者集合
private List<Observer> observers;
public float getTemperature() {
return temperature;
}
public float getPressure() {
return pressure;
}
public float getHumidity() {
return humidity;
}
public void setData(float temperature, float pressure, float humidity) {
this.temperature = temperature;
this.pressure = pressure;
this.humidity = humidity;
dataChange();
}
private void dataChange() {
notifyObservers();
}
public WeatherData() {
observers = new ArrayList<Observer>();
}
public void registerObserver(Observer o) {
observers.add(o);
}
public void removeObserver(Observer o) {
if (observers.contains(o)) {
observers.remove(o);
}
}
public void notifyObservers() {
observers.stream().forEach(a -> {
a.update(this.temperature, this.pressure, this.humidity);
});
}
}
3)观察者接口Observer:
public interface Observer {
public void update(float temperature, float pressure, float humidity);
}
4)具体观察者SinaSite:
public class SinaSite implements Observer {
private float temperature;
private float pressure;
private float humidity;
public void update(float temperature, float pressure, float humidity) {
this.temperature = temperature;
this.pressure = pressure;
this.humidity = humidity;
display();
}
public void display() {
System.out.println("----新浪网站----");
System.out.println("气温:" + temperature);
System.out.println("气压:" + pressure);
System.out.println("湿度:" + humidity);
}
}
5)调用:
public class Client {
public static void main(String[] args) {
WeatherData weatherData = new WeatherData();
//创建观察者
SinaSite sinaSite = new SinaSite();
//注册到weatherData
weatherData.registerObserver(sinaSite);
System.out.println("通知各个注册的观察者,看看信息");
weatherData.setData(20f, 100f, 30f);
}
}
- 观察者模式的好处:
- 观察者模式设计后,会以集合的方式来管理用户(Observer),包括注册,移除和通知。
- 这样,我们增加观察者(这里可以理解成一个新的公告板),就不需要去修改核心类WeatherData的代码,遵守了OCP原则。