观察者模式

Define a one-to-many dependency between objects where a state change in one object results in all its dependents being notified and updated automatically.

直译过来就是,定义对象间的一对多依赖关系,当一个对象的状态变更会自动通知和更新所有依赖项。像发布/订阅模式,事件通知模式,数据源/监听模式等都是性质一样。

观察者模式UML

UML类和时序图

类图:Subject类不会直接更新从属对象的状态。相反,Subject引用了用于更新状态的观察者接口(update()),这使得Subject独立于依赖对象的状态更新方式。Observer1和Observer2类通过将状态与Subject的状态同步来实现Observer接口。

时序图:Observer1和Observer2对象调用Subject1上的attach(this)来注册自己。假如Subject1的状态发生变更,Subject1本身调用notify()。notify()对已注册的Observer1和Observer2对象调用update(),后者从Subject1请求已更改的数据(getState())以更新(同步)其状态。

UML类图

观察者模式角色

Subject(目标):目标又称为主题,它是指被观察的对象。在目标中定义了一个观察者集合,一个观察目标可以接受任意数量的观察者来观察,它提供一系列方法来增加和删除观察者对象,同时它定义了通知方法notify()。目标类可以是接口,也可以是抽象类或具体类。

ConcreteSubject(具体目标):具体目标是目标类的子类,通常它包含有经常发生改变的数据,当它的状态发生改变时,向它的各个观察者发出通知;同时它还实现了在目标类中定义的抽象业务逻辑方法(如果有的话)。如果无须扩展目标类,则具体目标类可以省略。

Observer(观察者):观察者将对观察目标的改变做出反应,观察者一般定义为接口,该接口声明了更新数据的方法update(),因此又称为抽象观察者。

ConcreteObserver(具体观察者):在具体观察者中维护一个指向具体目标对象的引用,它存储具体观察者的有关状态,这些状态需要和具体目标的状态保持一致;它实现了在抽象观察者Observer中定义的update()方法。

观察者模式描述了如何建立对象与对象之间的依赖关系,以及如何构造满足这种需求的系统。观察者模式包含观察目标和观察者两类对象,一个目标可以有任意数目的与之相依赖的观察者,一旦观察目标的状态发生改变,所有的观察者都将得到通知。作为对这个通知的响应,每个观察者都将监视观察目标的状态以使其状态与目标状态同步,这种交互也称为发布-订阅(Publish-Subscribe)。观察目标是通知的发布者,它发出通知时并不需要知道谁是它的观察者,可以有任意数目的观察者订阅它并接收通知。

干货源码解析

源码地址

博客订阅的功能,抽象主题中维护订阅关系,同时引入普通和vip观察者。

//抽象主题
@Data
public abstract class Subject {//主题订阅者们private List<Observer> observerList = Lists.newArrayList();//订阅public void register(Observer observer) {observerList.add(observer);}//取消订阅public void remove(Observer observer) {observerList.remove(observer);}//发布东西public abstract void publish(String msg);
}//抽象观察者
@Slf4j
@Data
@AllArgsConstructor
public abstract class Observer {//观察者名称private String name;//更新状态,由主题调度public void update(Object subject, Object args) {log.info("{}获取到变更通知:{}", name, args);}
}//博客主题
@Slf4j
public class Blog extends Subject {@Overridepublic void publish(String msg) {log.info("发布msg:{}", msg);//通知订阅者getObserverList().forEach(observer -> observer.update(this, msg));}
}//普通用户观察者
@Slf4j
public class NormalObserver extends Observer {public NormalObserver(String name) {super(name);}@Overridepublic void update(Object subject, Object args) {super.update(subject, args);log.info("{}获取到变更通知:普通用户可以不缓存", getName());}
}//vip用户观察者
@Slf4j
public class VipObserver extends Observer {public VipObserver(String name) {super(name);}@Overridepublic void update(Object subject, Object args) {super.update(subject, args);log.info("{}获取到变更通知:vip可以缓存", getName());}
}@Slf4j
public class Application {public static void main(String[] args) {Blog blog = new Blog();VipObserver wang = new VipObserver("老王99");VipObserver lee = new VipObserver("老李");NormalObserver four = new NormalObserver("小四");NormalObserver twoEgg = new NormalObserver("二蛋");log.info("---------------------begin--------------------");// 用户订阅博客,普通和vip用户new Thread(() -> {blog.register(wang);sleep(2);blog.register(lee);sleep(2);blog.register(four);}).start();// 博客线程每隔2秒发布一次文章,  总共发布4次new Thread(() -> {IntStream.rangeClosed(1, 4).forEach(i -> {blog.publish(String.format("新把戏第%s次", i));sleep(2);});}).start();// 有用户退出订阅博客,也有二蛋加入订阅new Thread(() -> {sleep(3);blog.remove(lee);sleep(2);blog.register(twoEgg);}).start();}private static void sleep(int seconds) {try {TimeUnit.SECONDS.sleep(seconds);} catch (InterruptedException e) {log.error("error : ", e);}}
}
复制代码

jdk内置的Obverser和Observable

@Slf4j
public class ObserverApplication {public static void main(String[] args) {log.info("Enter Text: ");EventSource eventSource = new EventSource();eventSource.addObserver((obj, arg) -> {log.info("Received response: {}", arg);});eventSource.addObserver((obj, arg) -> {log.info("Received response2: {},,{}", arg, obj);});new Thread(eventSource).start();}
}class EventSource extends Observable implements Runnable {public void run() {while (true) {String response = new Scanner(System.in).next();setChanged();notifyObservers(response);}}
}
复制代码

Java中的使用

它使得一个对象可以灵活的将消息发送给感兴趣的对象

java.util.EventListener

javax.servlet.http.HttpSessionBindingListener

javax.servlet.http.HttpSessionAttributeListener

javax.faces.event.PhaseListener

Listener从名字上看就明白是监听的意思了。

JDK中内置的Obverser和Observable

jdk中内置的观察者模式。即是java.util.Observer(接口)和java.util.Observable(类)。

简单的说说这个。观察者接口(java.util.Observer),主题(java.util.Observable)。实现观察者接口和继承主题。通过抽象主题的addObserver()注册观察者,deleteObserver()移除观察者。

首先通过调用主题类的setChange()告知状态变更,随后调用notifyObservers方法(可传可不传参数)去通知观察者,最后因为主题在notifyObservers时会主动调用观察者的update()方法,改方法有2个参数,第一个为主题对象,第二个为可变参数。

总结

观察者模式是一种使用频率非常高的设计模式,JDK中就带有实现。无论是移动应用、Web应用或者桌面应用,观察者模式几乎无处不在,它为实现对象之间的联动提供了一套完整的解决方案,凡是涉及到一对一或者一对多的对象交互场景都可以使用观察者模式。观察者模式广泛应用于各种编程语言的GUI事件处理的实现,在基于事件的XML解析技术(如SAX2)以及Web事件处理中也都使用了观察者模式。

观察者模式的优点

(1) 观察者模式可以实现表示层和数据逻辑层的分离,定义了稳定的消息更新传递机制,并抽象了更新接口,使得可以有各种各样不同的表示层充当具体观察者角色。 (2) 观察者模式在观察目标和观察者之间建立一个抽象的耦合。观察目标只需要维持一个抽象观察者的集合,无须了解其具体观察者。由于观察目标和观察者没有紧密地耦合在一起,因此它们可以属于不同的抽象化层次。 (3) 观察者模式支持广播通信,观察目标会向所有已注册的观察者对象发送通知,简化了一对多系统设计的难度。 (4) 观察者模式满足“开闭原则”的要求,增加新的具体观察者无须修改原有系统代码,在具体观察者与观察目标之间不存在关联关系的情况下,增加新的观察目标也很方便。

观察者模式的缺点

(1) 如果一个观察目标对象有很多直接和间接观察者,将所有的观察者都通知到会花费很多时间。 (2) 如果在观察者和观察目标之间存在循环依赖,观察目标会触发它们之间进行循环调用,可能导致系统崩溃。 (3) 观察者模式没有相应的机制让观察者知道所观察的目标对象是怎么发生变化的,而仅仅只是知道观察目标发生了变化。

注意事项

1、JAVA 中已经有了对观察者模式的支持类。 2、避免循环引用。 3、如果顺序执行,某一观察者错误会导致系统卡壳,一般采用异步方式。

参考

Observer pattern

观察者模式|菜鸟教程

细数JDK里的设计模式

设计模式总结(Java)—— 观察者模式

欢迎关注

转载于:https://juejin.im/post/5caaa123e51d452b653f8f17

折腾Java设计模式之观察者模式相关推荐

  1. 折腾Java设计模式之建造者模式

    博文原址:折腾Java设计模式之建造者模式 建造者模式 Separate the construction of a complex object from its representation, a ...

  2. java 观察者模式_图解Java设计模式之观察者模式

    图解Java设计模式之观察者模式 天气预报项目需求 天气预报设计方案 1 - 普通方案 观察者模式(Observer)原理 观察者模式解决天气预报需求 观察者模式在JDK应用的源码分析 天气预报项目需 ...

  3. Java设计模式系列--观察者模式(使用)

    原文网址:Java设计模式系列--观察者模式(使用)_IT利刃出鞘的博客-CSDN博客 简介 说明 本文用示例介绍观察者模式的用法. 观察者模式的含义 以微信公众号为例.我们关注了某个微信公众号后能收 ...

  4. Java设计模式之观察者模式(UML类图分析+代码详解)

    大家好,我是一名在算法之路上不断前进的小小程序猿!体会算法之美,领悟算法的智慧~ 希望各位博友走过路过可以给我点个免费的赞,你们的支持是我不断前进的动力!! 加油吧!未来可期!! 本文将介绍java设 ...

  5. java设计模式 观察者模式_理解java设计模式之观察者模式

    在生活实际中,我们经常会遇到关注一个事物数据变化的情况,例如生活中的温度记录仪,当温度变化时,我们观察它温度变化的曲线,温度记录日志等.对于这一类问题,很接近java设计模式里面的"观察者模 ...

  6. Java 设计模式之观察者模式

    一.了解观察者模式 1.1 什么是观察者模式 观察者模式定义了对象之间的一对多依赖,这样一来,当一个对象状态改变时,它的所有依赖者都会收到通知并自动更新. 典型的问题比如报社,只要你是他们的订户,他们 ...

  7. [java设计模式简记] 观察者模式(Observer-Pattern)

    观察者模式(Observer-Pattern) 数据主体拥有需要数据的对象的数据,并且数据改变时需要数据的对象要及时知道 意图: 定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖 ...

  8. (设计模式十三)java设计模式之观察者模式

    特别说明:本篇博客来自于设计模式菜鸟教程 观察者模式 当对象间存在一对多关系时,则使用观察者模式(Observer Pattern).比如,当一个对象被修改时,则会自动通知它的依赖对象.观察者模式属于 ...

  9. Java设计模式:观察者模式

    观察者模式 观察者模式又称为发布/订阅(Publish/Subscribe)模式 在对象之间定义了一对多的依赖,这样一来,当一个对象改变状态,依赖它的对象会收到通知并自动更新. 如果这句话不好理解 可 ...

最新文章

  1. Netscaler 认证,访问报http 5000 内部错误
  2. mysql使用MRG_MyISAM(MERGE)实现水平分表
  3. 在linux上安装mysql5.6,在linux(Centos 7以上版本亲测)上安装mysql5.6
  4. Python遇到的问题(二)
  5. [导入]创建一个ASP通用分页类代码部分
  6. ffmpeg sdk java_推荐一个强大的音视频处理的开源项目!
  7. zabbix客户端部署
  8. Visual Studio中的项目属性--生成--配置
  9. 对Retinex算法的一些理解
  10. 抖音超强黑科技的到来真的有吗?
  11. 大型稀疏矩阵计算的现代方法介绍
  12. 计算机网络网线颜色,网络线接法-各种网线的分类和区别-电气资讯 - 电工屋
  13. 模电:集成运算放大器
  14. 锁定计算机后qq能远程吗,如何通过QQ进行远程控制(利用QQ远程协助在不同网络下远程控制电脑)...
  15. Facebook三方登陆流程
  16. 北京内推 | Hulu机器学习应用平台团队招聘推荐大数据方向暑期实习生
  17. ajax 传数据视频,Django知识整理四(choices参数,MTV与MVC模型,ajax介绍,ajax传json文件,ajax传文件,contentType前后端传输数据编码格式)...
  18. awk中RS,ORS,FS,OFS区别与联系
  19. http://39.98.219.132 题库标准答案(题库序号:463)之编写一个给一个分数约分的程序
  20. qq侧滑界面的实现(DreawerLayout)

热门文章

  1. 警惕 频繁染发伤肾脏
  2. 618什么数码好物值得买、2022值得买的数码好物指南
  3. 运动蓝牙耳机哪个品牌好、最适合运动的蓝牙耳机品牌
  4. 计算机网络术语英文翻译 (部分图片来源于网络,如侵权请通知)
  5. 2022年软考考试时间安排
  6. Delphi中画一个矩形
  7. html touchend 获取坐标,HTML touchend事件用法及代码示例
  8. PixInsight星空图片处理大致流程
  9. 常见Linux运维面试题分享
  10. 怎样录制电脑屏幕上播放的影视视频