观察者模式(Observer Pattern)
定义
- 观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态发生变化时,会通知所有观察者对象,使它们能够自动更新自己。
- 别名:模型-视图模式、源-收听者模式、从属者模式、发布-订阅模式。
- 观察者模式通过解耦观察者和被观察者,使得它们可以独立地变化和复用。
- 是行为型设计模式。
适用场景
- 当一个抽象模型包含两个方面内容,其中一个方面依赖于另一个方面;
- 其他一个或多个对象的变化依赖于另一个对象的变化;
- 实现类似于广播机制的功能,无需知道具体收听者,只需分发广播,系统中感兴趣的对象会自动接收该广播;
- 多层级嵌套使用,形成一种链式触发机制,使得事件具备跨域(跨越两种观察者类型)通知。
标准示例
观察者模式通常包含以下几个角色:
Subject(主题/被观察者):维护一个观察者列表,并提供注册、移除和通知观察者的方法。
Observer(观察者):定义一个更新接口,以便在得到主题通知时更新自己。
ConcreteSubject(具体主题):具体的被观察者,在内部状态改变时,通知所有注册的观察者。
ConcreteObserver(具体观察者):实现抽象观察者的定义,以便在得到主题状态更改通知时更新自身。
示例代码:
ISubject
抽象主题
/*** 抽象主题*/
public interface ISubject<E> {boolean attach(IObserver<E> observer);boolean detach(IObserver<E> observer);void notify(E event);
}
IObserver
抽象观察者
/*** 抽象观察者*/
public interface IObserver<E> {/*** 更新* @param event*/void update(E event);
}
ConcreteSubject
具体主题
/*** 具体主题*/
public class ConcreteSubject<E> implements ISubject<E>{private List<IObserver<E>> observers = new ArrayList<IObserver<E>>();public boolean attach(IObserver<E> observer) {return !this.observers.contains(observer) && this.observers.add(observer);}public boolean detach(IObserver<E> observer) {return this.observers.remove(observer);}public void notify(E event) {for(IObserver<E> observer:observers){observer.update(event);}}
}
ConcreteObserver
具体观察者
public class ConcreteObserver<E> implements IObserver<E>{public void update(E event) {System.out.println(this.getClass().getSimpleName() + " receive event : " + event);}
}
客户端调用代码:
public class ClientTest {public static void main(String[] args) {//被观察者ISubject<String> subject = new ConcreteSubject<String>();//观察者IObserver<String> observer = new ConcreteObserver<String>();//被观察者注册subject.attach(observer);//通知subject.notify("hello");}
}
输出执行结果:
ConcreteObserver receive event : hello
接下来,我们使用jdk中提供的观察者模式接口来实现一个如下场景:
明星发声,粉丝关注并收听。
Idol
明星主题抽象类
/*** 抽象主题者——偶像*/
public interface Idol {void posting(String message);
}
ConcreteIdol
明星主题具体类
/*** 具体偶像类*/
@Data
public class ConcreateIdol extends Observable implements Idol{private String name;public ConcreateIdol(String name){this.name = name;}public void posting(String message) {System.out.println("【明星发布】 "+message);setChanged();notifyObservers(message);}
}
Fans
粉丝具体类
/*** 观察者*/
public class Fans implements Observer {private String idol;private String selfName;public Fans(String idol,String selfName){this.idol = idol;this.selfName = selfName;}public void update(Observable o, Object arg) {System.out.println("【粉丝"+this.selfName+"收到通知】 " + "收到偶像消息:"+ arg);}
}
JDK中,java.util.Observable 类,相当于IObserver的存在,所以,我们的Fans类,只要继承Observable即可。在发布的被监听方法中,只要调用如下两个方法,可以实现发布。
setChanged();
notifyObservers();
而监听者实现类 Fans,只需要实现jdk中的 java.util.Observer 接口。其中的 update 方法,即为订阅到发布消息后的回调方法。
ClientTest
客户端调用类:
public class ClientTest {public static void main(String[] args) {ConcreateIdol zhazhahui = new ConcreateIdol("渣渣辉");zhazhahui.addObserver(new Fans(zhazhahui.getName(),"小明明"));zhazhahui.addObserver(new Fans(zhazhahui.getName(),"小甜甜"));String message = "大家好,我是"+zhazhahui.getName()+",欢迎体验我代言的新版砍怪看到手抽筋大型即使PK古装仙侠网游!";zhazhahui.posting(message);}
}
执行后的结果输出为:
【明星发布】 大家好,我是渣渣辉,欢迎体验我代言的新版砍怪看到手抽筋大型即使PK古装仙侠网游!
【粉丝小甜甜收到通知】 收到偶像消息:大家好,我是渣渣辉,欢迎体验我代言的新版砍怪看到手抽筋大型即使PK古装仙侠网游!
【粉丝小明明收到通知】 收到偶像消息:大家好,我是渣渣辉,欢迎体验我代言的新版砍怪看到手抽筋大型即使PK古装仙侠网游!