4.7 观察者模式
-
模式动机与定义
- 模式动机
- MVC示意图
- 软件系统: 一个对象的状态或行为的变化将导致其他的对象的状态或行为也发生改变,它们之间将产生联动
- 观察者模式:
- 定义了对象之间一种一对多的依赖关系,让一个对象的改变能够影响其他对象
- 发生改变的对象称为观察目标,被通知的对象称为观察者
- 一个观察目标可以对应多个观察者
- 模式定义
- 观察者模式(Observer Pattern):定义对象间的一种一对多依赖关系,使得每当一个对象状态发生改变时,其相关依赖对象皆得到通知并自动更新。
- 观察者模式又叫做发布-订阅(Publish/Subscriber)模式,模型-视图(Model/View)模式,源-监听器(Source/Listener)模式,或从属者(Dependents)模式
- 观察者模式是一种对象行为型模式
- 模式动机
-
模式结构与分析
- 模式结构
- 角色:
- Subject: 目标
- ConcreteSubject: 具体目标
- Observer: 观察者
- ConcreteObserver: 具体观察者
- 角色:
- 模式分析
- 说明:
- 有时候在具体观察者类ConcreteObserver中需要使用到具体目标类ConcreteSubject中的状态(属性),会存在关联或者依赖关系
- 如果在具体层之间具有关联关系,系统的扩展性将受到一定的影响,增加新的具体目标类有时候需要修改原有观察者的代码,在一定程度上违背了开闭原则,但是如果原有观察者类无须关联新增的具体目标,则系统扩展性不受影响
- 说明:
- 模式结构
模式实例与解析
-
模式效果与应用
-
模式优点
- 可以实现表示层和数据逻辑层的分离
- 在观察目标和观察者之间建立一个抽象的耦合
- 支持广播通信,简化了一对多系统设计的难度
- 符合开闭原则,增加新的具体观察者无须修改原有系统代码,在具体观察者与观察目标之间不存在关联关系的情况下,增加新的观察目标也很方便
-
模式缺点
- 将所有的观测者都通知到会花费很多时间
- 如果存在循环依赖时可能导致系统崩溃
- a调用b b里又调用a
- 没有相应的机制让观察者知道所观察的目标对象是怎么发生变化的,而只是知道观察目标发生了变化
-
模式应用
- 一个抽象模型有两个方面,其中一个方面依赖于另一方面,将这两个方面封装在独立对象中使它们可以各自独立地改变和复用
- 一个对象的改变将导致一个或多个其他对象发生改变,且并不知道具体有多少对象将发生改变,也不知道这些对象是谁
- 需要在系统中创建一个触发链
-
interface MyObsever {
public abstract void response();
}
import java.util.ArrayList;
abstract class MySubject {
protected ArrayList observers = new ArrayList();
//注册方法
public void attach(MyObsever obsever) {
observers.add(obsever);
}
//注销方法
public void detach(MyObsever obsever) {
observers.remove(obsever);
}
//通知方法
public abstract void cry();
}
class Cat extends MySubject {
@Override
public void cry() {
// TODO Auto-generated method stub
System.out.println("猫叫!");
System.out.println("-----------");
for(Object obs : observers) {
((MyObsever)obs).response();
}
}
}
class Dog implements MyObsever {
@Override
public void response() {
// TODO Auto-generated method stub
System.out.println("狗也跟着叫!");
}
}
public class Mouse implements MyObsever {
@Override
public void response() {
// TODO Auto-generated method stub
System.out.println("老鼠努力跑!");
}
}
public class Client {
public static void main(String[] args) {
// TODO Auto-generated method stub
MySubject cat = new Cat();
MyObsever mouse,dog;
mouse = new Mouse();
dog = new Dog();
cat.attach(mouse);
cat.attach(dog);
cat.cry();
}
}