一、模式示例
1.1、原理
1、定义
- 当对象间存在一对多关系时,则使用观察者模式(Observer Pattern)。比如,当一个对象被修改时,则会自动通知它的依赖对象。观察者模式属于行为型模式。
2、介绍
-
意图:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。
-
主要解决:一个对象状态改变给其他对象通知的问题,而且要考虑到易用和低耦合,保证高度的协作。
-
何时使用:一个对象(目标对象)的状态发生改变,所有的依赖对象(观察者对象)都将得到通知,进行广播通知。
-
如何解决:使用面向对象技术,可以将这种依赖关系弱化。
-
关键代码:在抽象类里有一个 ArrayList 存放观察者们。
-
应用实例:
1、拍卖的时候,拍卖师观察最高标价,然后通知给其他竞价者竞价。
2、西游记里面悟空请求菩萨降服红孩儿,菩萨洒了一地水招来一个老乌龟,这个乌龟就是观察者,他观察菩萨洒水这个动作。 -
优点:
1、观察者和被观察者是抽象耦合的。
2、建立一套触发机制。 -
缺点:
1、如果一个被观察者对象有很多的直接和间接的观察者的话,将所有的观察者都通知到会花费很多时间。
2、如果在观察者和观察目标之间有循环依赖的话,观察目标会触发它们之间进行循环调用,可能导致系统崩溃。
3、观察者模式没有相应的机制让观察者知道所观察的目标对象是怎么发生变化的,而仅仅只是知道观察目标发生了变化。 -
使用场景:
1、一个抽象模型有两个方面,其中一个方面依赖于另一个方面。将这些方面封装在独立的对象中使它们可以各自独立地改变和复用。
2、一个对象的改变将导致其他一个或多个对象也发生改变,而不知道具体有多少对象将发生改变,可以降低对象之间的耦合度。
3、一个对象必须通知其他对象,而并不知道这些对象是谁。
4、需要在系统中创建一个触发链,A对象的行为将影响B对象,B对象的行为将影响C对象……,可以使用观察者模式创建一种链式触发机制。 -
注意事项:
1、JAVA 中已经有了对观察者模式的支持类。
2、避免循环引用。
3、如果顺序执行,某一观察者错误会导致系统卡壳,一般采用异步方式。
3、UML图
4、角色
- 被观察者(Subject):当需要被观察部分的状态发生改变时,需要通知序列中所有的观察者;Subject需要维持(增、删、通知)一个观察者对象的列表
- 具体被观察者:被观察者的具体实现,包含一些基本的属性状态和其它的操作
- 观察者(Observer):接口或者抽象类;当Subject发生变化时,Observer对象将会通过一个函数得到通知
- 具体观察者:观察者的实现类,得到通知后执行一些具体的业务逻辑
5、需求
- 观察气象站的温度、湿度的变化
- java里面的Observable和observer接口可以直接实现观察者模式
1.2、java接口实现观察者模式
1、被观察者:气象站;实现Observable接口
public class MeteorologicalSubject extends Observable {
private double temperature;
private double humidity;
public double getTemperature() {
return temperature;
}
public void setTemperature(double temperature) {
this.temperature = temperature;
this.setChanged();
this.notifyObservers();
}
public double getHumidity() {
return humidity;
}
public void setHumidity(double humidity) {
this.humidity = humidity;
this.setChanged();
this.notifyObservers();
}
}
2、观察者;实现observer接口
public class MyObserver implements Observer {
@Override
public void update(Observable o, Object arg) {
System.out.println("气象站温度湿度已经发生改变");
}
}
3、启动类
public class ObserverMain {
public static void main(String[] args) {
MyObserver myObserver1 = new MyObserver();
MyObserver myObserver2 = new MyObserver();
MeteorologicalSubject meteorologicalSubject = new MeteorologicalSubject();
meteorologicalSubject.addObserver(myObserver1);
meteorologicalSubject.addObserver(myObserver2);
System.out.println("观察者数量 = "+meteorologicalSubject.countObservers());
meteorologicalSubject.setHumidity(34);
}
}
4、测试结果
- 如上可知:有两个观察者,所以当被观察者属性发生改变时,会打印两次推送消息
1.3、自定义观察者模式
1、被观察者:气象站
public class MeteorologicalSubject {
private double temperature;
private double humidity;
private List<Observer> observers = new ArrayList<>();
public double getTemperature() {
return temperature;
}
public void setTemperature(double temperature) {
this.temperature = temperature;
this.notifyObservers();
}
public double getHumidity() {
return humidity;
}
/*/**
* @Description :添加新的观察者
* @author : chuan
* @param : [observer]
* @return : void
* @exception :
*/
public void addObserver(Observer observer){
observers.add(observer);
}
/*/**
* @Description :获取所有观察者的数量
* @author : chuan
* @param : []
* @return : int
* @exception :
*/
public int getObserversSize(){
return observers.size();
}
public void setHumidity(double humidity) {
this.humidity = humidity;
this.notifyObservers();
}
/*/**
* @Description :消息推送:当属性改变时调用
* @author : chuan
* @param : []
* @return : void
* @exception :
*/
public void notifyObservers(){
for (Observer observer : observers) {
observer.update();
}
}
}
2、抽象观察者角色
public abstract class Observer {
public MeteorologicalSubject meteorologicalSubject;
abstract void update();
}
3、具体观察者:湿度观察者
public class HumidityObserver extends Observer{
public HumidityObserver(MeteorologicalSubject meteorologicalSubject) {
this.meteorologicalSubject = meteorologicalSubject;
}
@Override
void update() {
System.out.println("湿度发生了改变 = " + meteorologicalSubject.getHumidity());
}
}
4、具体观察者:温度观察者
public class TemperatureObserver extends Observer {
public TemperatureObserver(MeteorologicalSubject meteorologicalSubject) {
this.meteorologicalSubject = meteorologicalSubject;
}
@Override
void update() {
System.out.println("温度发生了改变 = " + meteorologicalSubject.getTemperature());
}
}
5、启动类
public class ObserverApplication {
public static void main(String[] args) {
MeteorologicalSubject meteorologicalSubject = new MeteorologicalSubject();
Observer temperatureObserver = new TemperatureObserver(meteorologicalSubject);
Observer humidityObserver = new HumidityObserver(meteorologicalSubject);
meteorologicalSubject.addObserver(temperatureObserver);
meteorologicalSubject.addObserver(humidityObserver);
meteorologicalSubject.setTemperature(23);
meteorologicalSubject.setHumidity(56);
}
}