观察者模式定义了对象间一对多的关系,当一个对象状态变化时,所有依赖它的对象都得到通知并将做出相应的反应。
Gof中的定义:观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态上发生变化时,会通知所有观察者对象,是他们能够自动更新。
应用场景:
1、监听器
2、SpringAOP切面
3、hook
4、callback
观察者UML类图如下
源事件(source)
源事件提供一个接口用来注册、删除、通知观察者角色,可以用一个抽象类或者接口来实现。具体的事件都应该继承或者实现源事件。
抽象观察者角色(Observer)
为所有观察者定义一个接口或者抽象类,在得到主题的通知时更新自己。通过接口封装监听器内部的实现细节
具体事件(ConcreteSource)
实现源事件,并重写其内部的方法,并在具体事件类中添加一个观察者集合,在注册或者删除一个观察者时是从集合中添加和删除的。这样在具体事件内部状态发生改变时,给所有观察它的观察者发出通知。
具体观察者(ConcreteObserver)
首先要实现观察者接口,然后将自身状态和事件相协调。如果需要观察者角色可以保存一个指向具体主题角色的引用。
举例:微信公众号推送
public interface Source {
void registerObserver(Observer o);
void unregisterObserver(Observer o);
void notifyObserver();
}
public interface Observer {
void update(String msg);
}
//具体事件
public class Financial implements Source {
private ArrayList<Observer> userList = new ArrayList<Observer>();
private String msg;
//在事件中注册一个监听器
@Override
public void registerObserver(Observer o) {
userList.add(o);
}
//卸载监听该事件的监听器集合中指定的监听器
@Override
public void unregisterObserver(Observer o) {
if (userList.contains(o)){
userList.remove(o);
}
}
//消息推送模型,通知监听器发生事件了
@Override
public void notifyObserver() {
for (Observer observer : userList) {
//监听器收到消息后采取不同的动作实现
observer.update(msg);
}
}
public void distributemsg(String s){
this.msg = s;
notifyObserver();
}
}
//观察者
public class User1 implements Observer{
public Source source;
//通过传参的方式将事件与监听器进行绑定,或者说是注册监听器
public User1(Source source){
this.source = source;
source.registerObserver(this);
}
@Override
public void update(String msg) {
System.out.println("我收到一个来自微信的理财消息" + msg + "非常不错");
}
}
public class User2 implements Observer {
public Source source;
public User2(Source source){
this.source = source;
source.registerObserver(this);
}
@Override
public void update(String msg) {
System.out.println("我对这个理财消息 " + msg + "不感兴趣");
}
}
测试程序
public class test {
public static void main(String[] args) {
Financial source = new Financial();
User1 user1 = new User1(source);
User2 user2 = new User2(source);
String msg = "支付宝理财";
//事件发生触发监听器
source.distributemsg(msg);
}
}
测试结果
我收到一个来自微信的理财消息支付宝理财非常不错
我对这个理财消息 支付宝理财不感兴趣
Process finished with exit code 0