不久之前,在学习 《SpringBoot 整合 WebSocket》 时,我们提到一个概念 —— 轮询
何为轮询?
轮询,就是在两点通讯的时候,客户端在固定的时间间隔下,不停的向服务端发送请求,查看服务端是否有最新的数据,如果有最新数据,则返回给客户端,如果没有,则返回空JSON。
轮询实现方便简单,但弊端也十分明显:服务端每次都要处理大量无效请求,在高并发场景下严重拖慢服务端运行效率,服务端资源也被极大的浪费。
为什么提到轮询呢?
因为轮询,是最方便实现两点之间通讯的方法。但效率太低。
那有没有一种方式,不需要不停的向数据源轮询,而是当有最新数据时,主动通知需求方呢?或者需求方时刻监听数据源数据变化,从而做出反应呢?
观察者模式
观察者模式又称为发布订阅模式,是对象的行为模式。
观察者模式定义一种一对多的依赖关系,让多个观察者对象同时监视着被观察者的状态,当被观察者的状态发生变化时,会通知所有观察者,并让其自动更新自己。
举个简单的例子:
小米官方旗舰店与一批电商平台合作,每次有新产品发布时,就推动到电商平台。
在这个例子中,小米旗舰店就是被观察对象,而电商平台(如淘宝、京东、拼多多)都是观察对象,当被监听对象发生变化时,监听对象就会触发指定行为。这就是观察者模式。
观察者模式可以根据被观察对象的变化做出指定行为,这种行为是独立的、责任明确的,也就是说,不同的观察者只需要维护自己的逻辑,而不会互相产生影响。
代码实例
-
定义被观察者对象
其中涉及到《单例设计模式》。Java 共有 23 中设计模式,单例设计模式是最基础、最简单的一种,这里不多做介绍,如有兴趣,可以前往 单例设计模式 了解。
package com.ambrose.observablepattern.observablepattern; import java.util.ArrayList; import java.util.List; import java.util.Observable; /** * 被观察者对象 * * 小米官方旗舰店产品列表 * * 被观察者对象,需要继承 java.util.Observable 类 * * 1、作为被观察者,要全局使用,且为了防止出现数据混乱,需要保证只有一个实例,故采用单例设计模式 * * */ public class XiaoMiProductList extends Observable { // 产品列表 private List<String> productList = null; /** * 单例模式部分 */ // 保存类的实例 private static XiaoMiProductList instance; // 私有化构造方法 private XiaoMiProductList(){} // 获取类的实例 public static XiaoMiProductList getInstance(){ if(instance == null) { instance = new XiaoMiProductList(); instance.productList = new ArrayList<String>(); } return instance; } /** * 观察者模式部分 */ /** * 增加观察者对象 * @param observable */ public void addObserver(Observable observable){ this.addObserver(observable); } public void addProduct(String product){ productList.add(product); System.out.println("小米商城新推出产品:" + product); // 设置被观察者对象为“改变”状态 this.setChanged(); // 通知观察者对象数据已发生变化,进行对应操作 this.notifyObservers(product); } }
-
定义观察者对象
package com.ambrose.observablepattern.observablepattern; import java.util.Observable; import java.util.Observer; /** * 观察者对象 * * 京东商城 * * 观察者对象,需要实现 java.util.Observer 接口 */ public class JDObserver implements Observer { @Override public void update(Observable o, Object arg) { String product = (String)arg; System.out.println( product + "已成功推送到【京东商城】"); } }
package com.ambrose.observablepattern.observablepattern; import java.util.Observable; import java.util.Observer; public class PDDObserver implements Observer { @Override public void update(Observable o, Object arg) { String product = (String) arg; System.out.println( product + "已成功推送到【拼多多商城】"); } }
package com.ambrose.observablepattern.observablepattern; import java.util.Observable; import java.util.Observer; public class TBObserver implements Observer { @Override public void update(Observable o, Object arg) { String product = (String) arg; System.out.println( product + "已成功推送到【淘宝商城】"); } }
-
测试观察者模式
package com.ambrose.observablepattern.observablepattern; import org.junit.Test; public class TestObservablePattern { @Test public void testObservablePattern(){ // 获取被观察者对象(单例模式) XiaoMiProductList observable = XiaoMiProductList.getInstance(); // 获取观察者对象 JDObserver jd = new JDObserver(); TBObserver tb = new TBObserver(); PDDObserver pdd = new PDDObserver(); // 被观察者对象中添加观察者对象 observable.addObserver(jd); observable.addObserver(tb); observable.addObserver(pdd); // 发布新产品 observable.addProduct("小米 10 Pro 至尊版"); observable.addProduct("Red mi K30 尊享版"); } }
-
测试结果