别再面向 for循环编程了,JDK 自带的观察者模式就很香

别再面向 for循环编程了,JDK 自带的观察者模式就很香

首页冒险解谜观察者系统重复更新时间:2024-05-07

作者:Java技术栈

来源:https://mp.weixin.qq.com/s/nEYhChMpLQhdZmTuo8jW6A

大家好,你还在面向 for 循环编程吗?

还有谁不会用观察者模式吗?

本篇带来《观察者模式》理论及实战~

什么是观察者模式?

观察者模式(Observer Pattern)定义了对象间的一种一对多的依赖关系,这样只要一个对象的状态发生改变,其依赖的所有相关对象都会得到通知并自动更新。

在观察者模式中,发生改变的对象叫做观察目标,而被通知更新的对象称为观察者,一个观察目标对应多个观察者,观察者一般是一个列表集合,可以根据需要动态增加和删除,易于扩展。

使用观察者模式的优点在于观察目标和观察者之间是抽象松耦合关系,降低了两者之间的耦合关系。

发布-订阅模式

观察者模式很多地方也叫发布-订阅模式(Publish/Subscribe),其实也可以这么理解,不过两者之间还是略有不同。

观察者模式中的观察者是直接绑定观察目标,观察目标要维护一套观察者列表,两者是有一个基于接口的组合依赖关系的,所以说观察者模式虽然是松耦合的,但并不是完全解耦的。

发布-订阅模式中的发布者和订阅者两者并没有任何联系,发布者通过中间方发布一个主题(Topic),订阅者通过中间方(调度中心)订阅一个主题(Topic),发布者状态的变更也并不会直接通知订阅者,而要通过中间方进行通知,或者订阅者自行从中间方拉取,所以说发布-订阅模式是完全解耦的。

一图搞懂它们的关系:

观察者模式和订阅发布模式的区别

从图片看两者是有差别的,统一都叫观察者模式,也没毛病。

观察者模式轮子

因观察者模式应用比较广泛,所以 JDK 工具包从 1.0 版本里面自带了观察者模式模板套装,我们根据其模板很方便就能实现观察者模式,不需要再重复造轮子了。

观察者目标类:

java.util.Observable

里面两个最重要的变量:

里面的重要的方法都是和观察目标状态和观察者相关的,一看就清楚,这里就不介绍了。

观察者接口:

java.util.Observable

publicinterfaceObserver{ /** *Thismethodiscalledwhenevertheobservedobjectischanged.An *applicationcallsan<tt>Observable</tt>object's *<code>notifyObservers</code>methodtohavealltheobject's *observersnotifiedofthechange. * *@paramotheobservableobject. *@paramarganargumentpassedtothe<code>notifyObservers</code> *method. */ voidupdate(Observableo,Objectarg); }

观察者接口只有一个 update 方法,用来通知观察者自己更新。

观察者模式实战

OK,知道了 JDK 自带了这两个东东,现在就来实现一个简单的观察者模式的应用场景,模拟公众号文章推送,观察目标是站长我,观察者是你们大家,我在公众号Java技术栈推一篇文章,你们都能接收到更新通知并能阅读。

新增观察目标类:

importlombok.Getter; importjava.util.Observable; /** *观察目标:栈长 *来源*Java技术栈 */ @Getter publicclassJavaStackObservableextendsObservable{ privateStringarticle; /** *发表文章 *@paramarticle */ publicvoidpublish(Stringarticle){ //发表文章 this.article=article; //改变状态 this.setChanged(); //通知所有观察者 this.notifyObservers(); } }

观察目标的逻辑是先发表文章,再改变观察目标的状态,再通知所有观察者。

我们来重点看 notifyObservers 方法的源码:

先获取同步锁,判断状态是否更新,如已更新则清空观察目标状态,然后再使用 for 循环遍历所有观察者,一一调用观察者的更新方法通知观察者更新。

新增观察者类:

importlombok.NonNull; importlombok.RequiredArgsConstructor; importjava.util.Observable; importjava.util.Observer; /** *观察者:读者粉丝 *来源*Java技术栈 */ @RequiredArgsConstructor publicclassReaderObserverimplementsObserver{ @NonNull privateStringname; privateStringarticle; @Override publicvoidupdate(Observableo,Objectarg){ //更新文章 updateArticle(o); } privatevoidupdateArticle(Observableo){ JavaStackObservablejavaStackObservable=(JavaStackObservable)o; this.article=javaStackObservable.getArticle(); System.out.printf("我是读者:%s,文章已更新为:%s\n",this.name,this.article); } }

观察者的逻辑是获取到观察者目标实例对象,然后再用观察目标对象的文章信息更新为自己的文章信息,最后输出某某某的文章已更新。

观察者只要实现 Observer 这个接口的 update 方法即可,用于观察目标进行调用通知。

本节教程所有实战源码已上传到这个仓库:https://github.com/javastacks/javastack

观察目标和观察者类结构图如下:

新增考试类:

/** *观察者:读者粉丝 *来源*Java技术栈 */ publicclassObserverTest{ publicstaticvoidmain(String[]args){ //创建一个观察目标 JavaStackObservablejavaStackObservable=newJavaStackObservable(); //添加观察者 javaStackObservable.addObserver(newReaderObserver("小明")); javaStackObservable.addObserver(newReaderObserver("小张")); javaStackObservable.addObserver(newReaderObserver("小爱")); //发表文章 javaStackObservable.publish("什么是观察者模式?"); } }

观察目标、观察者的创建并没有先后顺序要求,重点是发表文章通知观察者之前,观察目标要添加观察者列表这一步不能少。

输出结果:

通过这个简单的文章推送实践,大家应该对观察者模式有一个基本的认知了,在实际工作当中也可以有很多场景拿去用,就一对多的依赖关系都可以考虑使用观察者模式。

总结

不容易啊,陆陆续续又干了大半天,你学会观察者模式了吗?

观察者模式的优点是为了给观察目标和观察者解耦,而缺点也很明显,从上面的例子也可以看出,如果观察者对象太多的话,有可能会造成内存泄露。

另外,从性能上面考虑,所有观察者的更新都是在一个循环中排队进行的,所以观察者的更新操作可以考虑做成线程异步(或者可以使用线程池)的方式,以提升整体效率。

本节教程所有实战源码已上传到这个仓库:

https://github.com/javastacks/javastack

查看全文
大家还看了
也许喜欢
更多游戏

Copyright © 2024 妖气游戏网 www.17u1u.com All Rights Reserved