定义
观察者模式定义了对象之间的一对多依赖,当一个对象改变状态时,它的所有依赖者都会收到通知并自动更新。
关键字
- Observable
- 即被观察者,也可以被叫做主题(Subject)。通常有注册方法(register),取消注册方法(remove)和通知方法(notify)。
- Observer
- 即观察者,可以接收到主题的更新。当对某个主题感兴趣的时候需要注册自己,在不需要接收更新时进行注销操作。
例子与应用
- 用户从报社订阅报纸,报社和用户之间是一对多依赖,用户可以在报社订阅(register)报纸,报社可以把最新的报纸发给用户(notify),用户自动收到更新。在用户不需要的时候还可以取消注册(remove)。
- Android中的EventBus,Rxjava的实现都是基于观察者模式的思想。再比如回调函数:Android中对Button的点击监听等等。
- 观察者模式可以用来解耦
观察者模式代码实现
现在我们用代码来实现上面订阅报纸的例子:
- NewProvider作为对于报社的抽象,每隔两秒钟向用户发送报纸。
- User作为用户的抽象,可以收到报纸。
- NewsModel作为对报纸本身的抽象。
1 | /** |
1 | /** |
1 | /** |
1 | /** |
1 | public class NewsModel { |
1 | /** |
1 | user:0 receive news:title:1 content:1 |
JDK观察者模式API
在JDK的util包内Java为我们提供了一套观察者模式的实现,在使用的时候我们只需要继承Observable和Observer类即可。此外在JDK的实现中还增加了一个布尔类型的changed域,通过设置这个变量来确定是否通知观察者。
Demo1
1 | public class NewsProvider extends Observable { |
1 | public class User implements Observer { |
1 | public class Test { |
Demo2
1 | public class Publish extends Observable { |
1 | public class Subscribe implements Observer { |
1 | public class Test { |
回调函数与观察者模式
你到一个商店买东西,刚好你要的东西没有货,于是你在店员那里留下了你的电话,过了几天店里有货了,店员就打了你的电话,然后你接到电话后就到店里去取了货。
在这个例子里:
- 你的电话号码就叫回调函数;
- 你把电话留给店员就叫登记回调函数;
- 店里后来有货了叫做触发了回调关联的事件;
- 店员给你打电话叫做调用回调函数;
- 你到店里去取货叫做响应回调事件。
在Android中我们有一个常用的回调:对于View点击事件的监听。
通常在我们使用的时候是这样的:
1 | xxxView.setOnClickListener(new View.OnClickListener() { |
这样我们就注册好了一个回调函数。
我们可以在View的源码里发现这个接口:
1 | /** |
当你setClickListener的时候在View的源码中可以看到对本地OnClickListener的初始化:
1 | /** |
当你的点击到一个View后Android系统经过一系列的调用最后到了View的performClick方法中:
1 | /** |
就在这里,触发了你的onClick方法,然后执行方法体。
这里我们的被观察者就是View,他的注册方法(register)就是setOnClickListener(),通知方法就是performClick;而OnClickListener就是观察者。只不过这里的只能注册一个观察对象而已。