这是我参与8月更文挑战的第25天,活动详情查看:8月更文挑战
观察者模式
多个对象间存在一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新
结构
- 抽象主题(Subject)角色:也叫抽象目标类,它提供了一个用于保存观察者对象的聚集类和增加、删除观察者对象的方法,以及通知所有观察者的抽象方法。
- 具体主题(Concrete Subject)角色:也叫具体目标类,它实现抽象目标中的通知方法,当具体主题的内部状态发生改变时,通知所有注册过的观察者对象。
- 抽象观察者(Observer)角色:它是一个抽象类或接口,它包含了一个更新自己的抽象方法,当接到具体主题的更改通知时被调用。
- 具体观察者(Concrete Observer)角色:实现抽象观察者中定义的抽象方法,以便在得到目标的更改通知时更新自身的状态。
演示
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、观察者模式只有观察者和被观察者;发布订阅模式除了观察者和被观察者,还有一个消息通道