观察者模式

371 阅读2分钟

这是我参与8月更文挑战的第25天,活动详情查看:8月更文挑战

观察者模式

多个对象间存在一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新

结构

  1. 抽象主题(Subject)角色:也叫抽象目标类,它提供了一个用于保存观察者对象的聚集类和增加、删除观察者对象的方法,以及通知所有观察者的抽象方法。
  2. 具体主题(Concrete Subject)角色:也叫具体目标类,它实现抽象目标中的通知方法,当具体主题的内部状态发生改变时,通知所有注册过的观察者对象。
  3. 抽象观察者(Observer)角色:它是一个抽象类或接口,它包含了一个更新自己的抽象方法,当接到具体主题的更改通知时被调用。
  4. 具体观察者(Concrete Observer)角色:实现抽象观察者中定义的抽象方法,以便在得到目标的更改通知时更新自身的状态。

image.png

演示

1、抽象主题

public abstract class Subject {
    protected List<Observer> list = new ArrayList<>();
    public void add(Observer observer) {
        if(!this.list.contains(observer)) {
            this.list.add(observer);
        }
    }
    public void delete(Observer observer) {
        this.list.remove(observer);
    }
    public abstract void notifyObserver();
}

2、具体主题

public class SubjectImpl extends Subject {
    @Override
    public void notifyObserver() {
        System.out.println("主题修改!");
        for (Observer observer : list) {
            observer.update();
        }
    }
}

3、抽象观察者

public abstract class Observer {
    public abstract void update();
}

4、具体观察者

public class ObserverOne extends Observer {
    @Override
    public void update() {
        System.out.println( this.getClass().getSimpleName() +  " 接收消息!");
    }
}
public class ObserverTwo extends Observer {
    @Override
    public void update() {
        System.out.println( this.getClass().getSimpleName() +  " 接收消息!");
    }
}

5、客户端

public class Client {
    public static void main(String[] args) {
        SubjectImpl subjectImpl = new SubjectImpl();
        subjectImpl.add(new ObserverOne());
        subjectImpl.add(new ObserverTwo());
        subjectImpl.notifyObserver();
        // 主题修改!
        // ObserverOne 接收消息!
        // ObserverTwo 接收消息!
    }
}

总结

优点

  1. 降低了目标与观察者之间的耦合关系,两者之间是抽象耦合关系。符合依赖倒置原则。
  2. 目标与观察者之间建立了一套触发机制。

缺点

  1. 目标与观察者之间的依赖关系并没有完全解除,而且有可能出现循环引用。
  2. 当观察者对象很多时,通知的发布会花费很多时间,影响程序的效率。

观察者模式和发布订阅模式的区别

有些将观察者模式称为发布订阅模式,个人观点,虽然目的和作用是基本一致的,但是这两者还是有些区别的

1、观察者模式虽然也有解耦的作用,但是并没有完成解耦,被观察者需要主动调用观察者的方法;而发布订阅这两者之间是不存在关系,观察者是通过第三方来通知的。

2、观察者模式只有观察者和被观察者;发布订阅模式除了观察者和被观察者,还有一个消息通道