观察者模式是对象的行为模式,又叫发布-订阅模式,模型-视图模式,源-监听器模式或者从属者模式。观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己。(解释引用于-java与模式)
package Observer; /* * @auther 顶风少年 * @mail dfsn19970313@foxmail.com * @date 2019-12-17 14:56 * @notify 模拟客户端 * @version 1.0 */ public class Client { //如果在spring项目中,我们可以把subject注册到容器中。当前做法仅想表示初始化一次。 static OrderSubject subject = null; static { subject = new OrderSubject(); subject.attach(new ChangeAccount()); subject.attach(new ChangeInventory()); } public static void main(String[] args) { //用户购买商品,付款成功 Order order = new Order("张三", "苹果"); //调用所有监听者,执行各自操作 subject.notifyObservers(order); } }
业务pojo
package Observer; /* * @auther 顶风少年 * @mail dfsn19970313@foxmail.com * @date 2019-12-17 14:45 * @notify 用于模拟订单信息 * @version 1.0 */ public class Order { public Order(String userId, String goodsId) { this.userId = userId; this.goodsId = goodsId; } //用户id private String userId; //商品id private String goodsId; public String getUserId() { return userId; } public void setUserId(String userId) { this.userId = userId; } public String getGoodsId() { return goodsId; } public void setGoodsId(String goodsId) { this.goodsId = goodsId; } }
订阅主题
package Observer; import java.util.ArrayList; import java.util.List; /* * @auther 顶风少年 * @mail dfsn19970313@foxmail.com * @date 2019-12-17 10:49 * @notify 订单主题 * @version 1.0 */ public class OrderSubject { private List<Observer> observersList = new ArrayList<>(); //调用这个方法登记一个新的观察者对象 public void attach(Observer observer) { observersList.add(observer); } //调用这个方法删除一个已经登记过的观察者对象 public void detach(Observer observer) { observersList.remove(observer); } //调用这个方法通知所有登记过的观察者对象 public void notifyObservers(Order order) { for (Observer observer : observersList) { observer.excuse(order); } } }
观察者接口
package Observer; /* * @auther 顶风少年 * @mail dfsn19970313@foxmail.com * @date 2019-12-17 10:56 * @notify 观察者接口 * @version 1.0 */ public interface Observer { //执行具体的业务 void excuse(Order order); }
账户余额业务观察者实现
package Observer; /* * @auther 顶风少年 * @mail dfsn19970313@foxmail.com * @date 2019-12-17 14:51 * @notify 修改账户余额 * @version 1.0 */ public class ChangeAccount implements Observer { //执行具体的业务 public void excuse(Order order) { String userId = order.getUserId(); System.out.println("正在修改用户:" + userId + "的账户余额"); System.out.println("账户余额修改成功"); } }
商品库存业务观察者实现
package Observer; /* * @auther 顶风少年 * @mail dfsn19970313@foxmail.com * @date 2019-12-17 14:51 * @notify 修改库存 * @version 1.0 */ public class ChangeInventory implements Observer { //执行具体的业务 public void excuse(Order order) { String goodsId = order.getGoodsId(); System.out.println("正在修改商品:" + goodsId + "的库存数量"); System.out.println("库存数量修改成功"); } }
平时写crud时,会经常遇到上边的业务情况。当一个对象改变,往往需要修改与之相关联的其他几个对象,这里不如说修改其他表更为准确。上边例子,常规来讲我们在用户付款成功后,依次调用扣款接口,库存接口来进行相应的操作。这其实增强了,支付业务和扣款业务,库存业务的耦合性。使用观察者模式,支付业务无需知道用户付款成功后,需要做什么操作,只需要找到主题,然后将消息发布出去即可。其次,我们可以将不同的业务放到不同的订阅组中,观察者和观察者之间没有关联,订阅组和订阅组之间也没有关联。
观察者模式的缺点也很明显,当一个活动对象有多个观察者,多个观察者循环执行,如果一个观察者出错,就可能造成整个生态的崩溃。