介绍

当对象间存在一对多关系时,则使用观察者模式(Observer Pattern)。比如,当一个对象被修改时,则会自动通知依赖它的对象。观察者模式属于行为型模式。

在GUI编程中这个模式很常见,数据更新后视图也会跟着改变
比如JavaFX里面的 javafx.collections.ObservableList

优缺点

优点:

  1. 观察者和被观察者是抽象耦合的。
  2. 建立一套触发机制。

缺点:

  1. 如果一个被观察者对象有很多的直接和间接的观察者的话,将所有的观察者都通知到会花费很多时间。
  2. 如果在观察者和观察目标之间有循环依赖的话,观察目标会触发它们之间进行循环调用,可能导致系统崩溃。
  3. 观察者模式没有相应的机制让观察者知道所观察的目标对象是怎么发生变化的,而仅仅只是知道观察目标发生了变化。

注意事项:

  1. JAVA 中已经有了对观察者模式的支持类。
  2. 避免循环引用。
  3. 如果顺序执行,某一观察者错误会导致系统卡壳,一般采用异步方式。

使用场景

  • 一个抽象模型有两个方面,其中一个方面依赖于另一个方面。将这些方面封装在独立的对象中使它们可以各自独立地改变和复用。
  • 一个对象的改变将导致其他一个或多个对象也发生改变,而不知道具体有多少对象将发生改变,可以降低对象之间的耦合度。
  • 一个对象必须通知其他对象,而并不知道这些对象是谁。
  • 需要在系统中创建一个触发链,A对象的行为将影响B对象,B对象的行为将影响C对象……,可以使用观察者模式创建一种链式触发机制。

实现

public interface Observer {
    void onPriceUpdate(int price);
}
public class Subject {
    int price = 10;
    List<Observer> list = new ArrayList<>();

    public int getPrice() {
        return price;
    }

    public void setPrice(int price) {
        this.price = price;
        notifyAllObserver();
    }

    private void notifyAllObserver() {
        for (Observer observer : list) {
            observer.onPriceUpdate(price);
        }
    }

    public void registerPriceObserver(Observer observer) {
        list.add(observer);
    }

    public void removePriceObserver(Observer o) {
        list.remove(o);
    }
}
public class AppTest {
    public static void main(String[] args) {
        Subject subject = new Subject();
        subject.registerPriceObserver(System.out::println);
        subject.registerPriceObserver(o -> System.out.println("我是2:" + o));
        subject.setPrice(3);
        subject.setPrice(4);
    }
}
博客
分类
标签
归档
关于