前言:
在现实世界中,许多对象并不是独立存在的,其中一个对象的行为发生改变可能会导致一个或者多个其他对象的行为也发生改变。例如,某种商品的物价上涨时会导致部分商家高兴,而消费者伤心;还有,当我们开车到交叉路口时,遇到红灯会停,遇到绿灯会行。这样的例子还有很多,例如,股票价格与股民、微信公众号与微信用户、气象局的天气预报与听众、小偷与警察等。
在软件世界也是这样,例如,Excel 中的数据与折线图、饼状图、柱状图之间的关系;MVC 模式中的模型与视图的关系;事件模型中的事件源与事件处理者。所有这些,如果用观察者模式来实现就非常方便。
定义与特点:
观察者(Observer)模式的定义:指多个对象间存在一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。这种模式有时又称作发布-订阅模式、模型-视图模式,它是对象行为型模式。
缺点:
-
降低了目标与观察者之间的耦合关系,两者之间是抽象耦合关系。符合依赖倒置原则。
-
目标与观察者之间建立了一套触发机制。
结构与实现:
举例:
关系图:
代码实现:
//抽象目标
public interface Subject {
void registerObserver(Observer o);
void removeObserver(Observer o);
void notifyObservers();
}
//抽象观察者
public interface Observer {
void update(float temperature,float pressure,float humidity);
}
//具体目标
public class WeatherData implements Subject {
private float temperature;
private float pressure;
private float humidity;
private ArrayList<Observer> al;
public WeatherData() {
this.al = new ArrayList<Observer>();
}
public float getTemperature() {
return temperature;
}
public float getPressure() {
return pressure;
}
public float getHumidity() {
return humidity;
}
public ArrayList<Observer> getAl() {
return al;
}
//推送信息
public void dataChange() {
notifyObservers();
}
//当数据有更新时,就调用setData方法,改变本目标数据
public void setData(float temperature, float pressure, float humidity) {
this.temperature = temperature;
this.pressure = pressure;
this.humidity = humidity;
dataChange();
}
//添加观察者
@Override
public void registerObserver(Observer o) {
al.add(o);
}
//删除观察者
@Override
public void removeObserver(Observer o) {
al.remove(o);
}
//遍历观察者集合,并通知
@Override
public void notifyObservers() {
for (int i = 0; i < al.size(); i++) {
al.get(i).update(this.temperature, this.pressure, this.humidity);
}
}
}
//具体观察者
public class CurrentConditions implements Observer {
//温度、气压、湿度
private float temperature;
private float pressure;
private float humidity;
public float getTemperature() {
return temperature;
}
public float getPressure() {
return pressure;
}
public float getHumidity() {
return humidity;
}
//更新天气情况,由WeatherData调用,使用推送模式
@Override
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("today temperature:" + getTemperature());
System.out.println("today pressure:" + getPressure());
System.out.println("today humidity:" + getHumidity());
}
}
//测试
public class Test {
public static void main(String[] args) {
//创建具体目标
WeatherData wd = new WeatherData();
//创建具体观察者
CurrentConditions cc = new CurrentConditions();
//注册到具体目标
wd.registerObserver(cc);
//测试
wd.setData(31f,4f,5f);
}
}
运行结果:
today temperature:31.0
today pressure:4.0
today humidity:5.0