1、基本概念
观察者模式( Observer Pattern ),又叫发布-订阅( Publish/Subscribe )模式、源-监听器( Source/Listener )模式。定义一
种一对多的依赖关系, 一个主题对象可被多个观察者对象同时监听,使得每当主题对象状态变化时, 所
有依赖于它的对象都会得到通知并被自动更新。属于行为型模式。
应用场景:微信公众号关注后的消息通知,app的消息通知,app的推送消息等
2、类图和角色
类图:
角色:
抽象的主题:即抽象的被观察者ISubject,定义了增加,删除,通知观察者的方法
具体的主题:即具体的被观察者ConcreteSubject,具体的实现,当发生变化时,会通知观察者
抽象的观察者:IObserver,定义响应通知的方法
具体的观察者:ConcreteObserver,定义了具体的响应通知方法,对被观察者做出相应的反应
3、案例
案例:app推送消息
类图:
JDK提供了自带的实现方式来帮助我们更方便的实现观察者模式
我们的被观察者只需要继承Observable类即可
具体的被观察者:BoringApp
public class BoringApp extends Observable {
public static final String APP_NAME = "无聊app";
public void pushMessage(Message msg){
System.out.printf("无聊app推送了一条消息:发送人:%s,消息内容%s \n ——————\n",msg.getUsername(),msg.getContent());
setChanged();
notifyObservers(msg);
}
}
我们的观察者只需要实现Observer接口
具体的观察者:User
@Data
public class User implements Observer {
private String name;
public User(String name){
this.name = name;
}
@Override
public void update(Observable o, Object arg) {
Message message = (Message)arg;
System.out.printf("【%s】, 你收到了来自%s的一条信息:%s\n-----\n", BoringApp.APP_NAME,message.getUsername(),message.getContent());
}
}
自定义消息类:
@Data
public class Message {
private String username;
private String content;
public Message(String usename,String content){
this.username = usename;
this.content = content;
}
}
测试:
public class Test {
public static void main(String[] args) {
BoringApp socalApp = new BoringApp();
//添加观察者
socalApp.addObserver(new User("老王"));
socalApp.addObserver(new User("老李"));
socalApp.pushMessage(new Message("小姐姐","明天去打篮球"));
}
}
输出结果:
【无聊app】推送了一条消息:发送人:小姐姐,消息内容明天去打篮球
——————
【无聊app】, 你收到了来自小姐姐的一条信息:明天去打篮球
-----
【无聊app】, 你收到了来自小姐姐的一条信息:明天去打篮球
-----
4、优缺点
优点:
1、观察者和被观察者是松耦合(抽象耦合)的,符合依赖倒置原则;
2、 分离了表示层(观察者)和数据逻辑层(被观察者) ,并且建立了一套触发机制,使得数据的变
化可以响应到多个表示层上;
3、实现了一对多的通讯机制,支持事件注册机制, 支持兴趣分发机制,当被观察者触发事件时,只
有感兴趣的观察者可以接收到通知。
缺点:
1、如果观察者数量过多,则事件通知会耗时较长;
2、事件通知呈线性关系,如果其中一个观察者处理事件卡壳,会影响后续的观察者接收该事件;
3、如果观察者和被观察者之间存在循环依赖,则可能造成两者之间的循环调用,导致系统崩溃。