概念
观察者模式(别名:依赖,发布-订阅)
定义对象间的一种一对多的依赖关系,当一个对象的状态发生变化时,所有依赖它的对象都得到通知并被自动更新Observer Pattern
Define a one-to-many dependency between objects so that when one object changes state, all its dependents are notified and updated automatically.
角色
- 主题(Subject):主题是一个接口,该接口规定了具体主题需要实现的方法。
- 观察者(Observer):观察者是一个接口,该接口规定了具体观察者用来更新数据的方法。
- 具体主题(ConcreteSubject):具体主题是实现主题接口类的一个实例,该实例包含有可以经常发生变化的数据。具体主题需要使用一个集合,例如ArrayList来存放观察者的引用,以便数据变化时通知具体观察者。
- 具体观察者(ConcreteObserver):具体观察者时实现观察者接口类的一个实例。
观察者模式的UML类图
观察者模式代码示例
详见Gitee仓库Java设计模式的Observer包
update()方法是如何实现的?
在求职中心的问题中,在Observer接口里,定义hearTelephone()
方法,然后要求具体的观察者对象实现hearTelephone()
方法来更新数据
public void hearTelephone(String heardMess);
notifyObserver()方法是如何传递的?
具体的主题里有一个列表,用来保存所有的观察者对象 personList: ArrayList<Observer>
每当具体主题的信息发生变化时,会遍历观察者列表,通知每一个观察者。
public void notifyObserver() {
if (changed) {
for (int i = 0; i < personList.size(); i ++) {
Observer observer = personList.get(i);
observer.hearTelephone(mess);
}
changed = false;
}
}
推数据和拉数据
update()
方法带参数,通过参数来传递数据,这样的方法叫推数据。
例如书上的求职中心例子中,通过hearTelephone(String heardMess)
来传递数据
同时,用update()
方法来传递数据的话,具体观察者对象中,就不用具体主题对象的引用。
拉数据提供获取这些数据的方法,具体观察者得到通知后,可以调用具体主题的方法得到数据,但需要自己判断数据是否发生了变化
拉数据相较于推数据的区别
update()
方法无参数- 在主题对象里,定义
getState()
和setState()
方法供观察者调 - 需要保存主题对象的引用
观察者模式的优点
- 具体主题和具体观察者是松耦合关系:具体主题里的观察者列表里保存的是抽象的观察者接口,而不是具体的观察者。
- 满足开闭原则:对扩展开放,对修改关闭。具体观察者对象增加一个具体主题后,已经存在的主题和观察者都不会修改。