一、观察者模式的定义:
简单地说,观察者模式定义了一个一对多的依赖关系,让一个或多个观察者对象监听一个主题对象。这样一来,当被观察者状态发生改变时,需要通知相应的观察者,使这些观察者对象能够自动更新。
观察者模式多用于实现订阅功能的场景,例如微博的订阅,当我们订阅了某个人的微博账号,当这个人发布了新的消息,就会通知我们。
二、观察者模式的实现:
1、Subject(被观察的对象接口):规定ConcreteSubject的统一接口 ; 每个Subject可以有多个Observer;
2、ConcreteSubject(具体被观察对象):维护对所有具体观察者的引用的列表 ;–状态发生变化时会发送通知给所有注册的观察者。
3、Observer(观察者接口):规定ConcreteObserver的统一接口;定义了一个update()方法,在被观察对象状态改变时会被调用。
4、ConcreteObserver(具体观察者):维护一个对ConcreteSubject的引用;特定状态与ConcreteSubject同步; 实现Observer接口,update()方法的作用:一旦检测到Subject有变动,就更新信息。
图表如下:
三、举例说明
有一个微信公众号服务,不定时发布一些消息,关注公众号就可以收到推送消息,取消关注就收不到推送消息。
四、观察者模式具体实现
1、定义一个抽象被观察者接口
package com.boss.info.observer;
/**
* Created by yd on 2019/3/20.
*/
public interface Observerable {
//注册为一个观察者
void addObserver(Observer o);
//取消观察者
void removeObserver(Observer o);
//通知所有观察者更新信息
void notifyObserver();
}
2、定义一个抽象观察者接口
package com.boss.info.observer;
/**
* 抽象观察者
* 定义了一个update()方法,当被观察者调用notifyObservers()方法时,观察者的update()方法会被回调。
* Created by yd on 2019/3/20.
*/
public interface Observer {
void update(String message);
}
3、定义被观察者,实现了Observerable接口,对Observerable接口的三个方法进行了具体实现,同时有一个List集合,用以保存注册的观察者,等需要通知观察者时,遍历该集合即可。
package com.boss.info.observer;
import lombok.extern.log4j.Log4j;
import java.util.ArrayList;
import java.util.List;
/**
* 被观察者,也就是微信公众号服务
* 实现了Observerable接口,对Observerable接口的三个方法进行了具体实现
* Created by yd on 2019/3/20.
*/
@Log4j
public class WebChat implements Observerable{
private List<Observer> list;
private String message;
public WebChat(){
list = new ArrayList<>();
}
@Override
public void addObserver(Observer o){
list.add(o);
}
@Override
public void removeObserver(Observer o){
if(!list.isEmpty()){
list.remove(o);
}
}
@Override
public void notifyObserver(){
for(Observer observer:list){
observer.update(message);
}
}
public void setInfo(String a){
this.message = a;
log.info("微信公众号更新,秒杀价格为"+a);
notifyObserver();
}
}
4、定义具体观察者,微信公众号的具体观察者为用户User
package com.boss.info.observer;
import lombok.extern.log4j.Log4j;
/**
* 观察者
* 实现了update方法
* @author yd
*/
@Log4j
public class User implements Observer{
private String name;
private String message;
public User(String name){
this.name = name;
}
@Override
public void update(String message){
this.message = message;
read();
}
public void read() {
log.info("恭喜你:"+name+",你已经获得秒杀资格,价格为"+message+"元");
}
}
5、编写一个测试类
首先有3个用户注册,并都接收到了信息;然后用户0003取消关注,则她再也收不到信息
package com.boss.info.observer;
/**
* Created by yd on 2019/3/20.
*/
public class Test {
public static void main(String[] args) {
WebChat w = new WebChat();
User a = new User("0001");
User b = new User("0002");
User c = new User("0003");
w.addObserver(a);
w.addObserver(b);
w.addObserver(c);
w.setInfo("1999.00");
w.removeObserver(c);
w.setInfo("999.00");
}
}
结果:
14:03:35.472 [main] INFO com.boss.info.observer.WebChat - 微信公众号更新,秒杀价格为1999.00
14:03:35.477 [main] INFO com.boss.info.observer.User - 恭喜你:0001,你已经获得秒杀资格,价格为1999.00元
14:03:35.477 [main] INFO com.boss.info.observer.User - 恭喜你:0002,你已经获得秒杀资格,价格为1999.00元
14:03:35.477 [main] INFO com.boss.info.observer.User - 恭喜你:0003,你已经获得秒杀资格,价格为1999.00元
14:03:35.477 [main] INFO com.boss.info.observer.WebChat - 微信公众号更新,秒杀价格为999.00
14:03:35.477 [main] INFO com.boss.info.observer.User - 恭喜你:0001,你已经获得秒杀资格,价格为999.00元
14:03:35.477 [main] INFO com.boss.info.observer.User - 恭喜你:0002,你已经获得秒杀资格,价格为999.00元
六、小结
这个模式是松偶合的。改变主题或观察者中的一方,另一方不会受到影像。
JDK中也有自带的观察者模式。但是被观察者是一个类而不是接口,限制了它的复用能力。
在JavaBean和Swing中也可以看到观察者模式的影子。