观察者模式

一)何为观察者模式
先来看看观察者模式的定义吧:对象之间一对多的依赖,这样一来,当一个对象(subject)改变状态时,它的所有依赖者(observer)都会收到通知并自动更新。
一个经典的例子便是: 出版者 + 订阅者 = 观察者模式。 如果你了解报纸订阅是怎么回事,其实就知道观察者模式是怎么回事。只是名称不太一样,出版者改称为subject, 订阅者改称为observer。对象只要订阅(add),就可以成为订阅者(observer),一旦出版者(subject)有新消息发布,订阅者就会知道。当然,当一个订阅者不再想订阅时,只需退订(remove)即可。
让我们先来看看一个已经实现的观察者模式是怎么样的:
 public class WeatherStation { //以气象站为例
public static void main(String[] args) {
WeatherData weatherData = new WeatherData(); //气象数据作为subject,一旦有新数据,则告知所有观察者。

CurrentConditionDisplay currentDisplay = new CurrentConditionDisplay(weatherData); //currentDisplay 注册成为observer
StatisticsDisplay statisticsDisplay = new StatisticsDisplay(weatherData); //statisticsDisplay注册成为observer
ForecastDisplay forecastDisplay = new ForecastDisplay(weatherData); //forecastDisplay 注册成为observer

weatherData.setMeasurements(80,65,30.4f); //subject有新数据,告知所有observer
weatherData.setMeasurements(82,70,29.2f); //subject有新数据,告知所有observer
weatherData.setMeasurements(78,90,29.2f); //subject有新数据,告知所有observer
}
}

看看以上程序运行的结果:

[img]http://dl.iteye.com/upload/attachment/0063/5601/51d87134-9741-3fa1-9e0d-312d2dc531b9.jpg[/img]

currentDisplay statisticsDisplay forecastDisplay 对每一次subject的改变都做出了相应的数据展示的调整。而为何每次[i]weatherData.setMeasurements(...)[/i] 那3个observer都能自动收到消息并作出反应呢?这便是观察者模式的威力所在噢~~ 想知道怎么做到的?接着看下去吧。

二)观察者模式的实现
仍以上述气象站为例,先来看看总体类关系结构:
[img]http://dl.iteye.com/upload/attachment/0063/5642/0fff8a9c-bd69-35a6-a609-78c0978860df.jpg[/img]

再看看主题(subject)如何实现:
[img]http://dl.iteye.com/upload/attachment/0063/5646/bfe848bb-43ac-3a25-9cc9-e38008290152.jpg[/img]

仅以CurrentConditionDisplay 为例,看看observer如何实现:
[img]http://dl.iteye.com/upload/attachment/0063/5656/87c581e3-a201-38b6-ba5b-889840230660.jpg[/img]

好了,一个观察者模式建好了!!很简单是不是? 但是这只是观察者模式的一种哦,我们是用subject将信息“推”给observer的方式实现的。相反,我们还可以让observer从subject中“拉”出信息的方式来实现观察者模式。以下一个例子在介绍Java内置观察者模式的同时,也会展示什么是“拉”的观察者模式。

三) Java内置观察者模式
java.util包内包含了最基本的Observable[b]类[/b](等效于Subject)和Observer接口(等效于Observer)。它们在使用上更为方便,因为许多功能都已经准备好了。你可以任意使用推或拉的方式传送数据。先来看看如果使用内置观察者模式,我们的类结构应该是怎样的:
[img]http://dl.iteye.com/upload/attachment/0063/5683/9ea6dfdf-1ff6-316f-8bf6-8ee87342bc75.jpg[/img]

先忽略setChange(),看看subject如何实现:
[img]http://dl.iteye.com/upload/attachment/0063/5692/61acee58-6c8c-3770-8b85-3e60c31a7f2f.jpg[/img]
注意:如果是拉,notifyObserver()应该删去。


再看看observer如何实现:
[img]http://dl.iteye.com/upload/attachment/0063/5697/1747f421-fe49-3161-b024-ecb1fcac4308.jpg[/img]

好了,这就是java内置的拉的观察者模式的全部了。剩下的就是解释下这个setChange()了。它是一个用来标记状态已改变的事实的标记位,好让notifyObservers()知道当它被调用时应该更新观察者。如果调用notifyObservers()之前没有先setChanged(),那么观察者就“不会”被通知。这样做的必要性在于它可以使你在更新观察者时有更多的弹性,你可以更适当的通知观察者。比如,我们的气象站测量是如此的敏锐以致于温度计读数每十分之一度的变化就会更新,这会造成WeatherData对象持续不断地通知观察者。如果我们希望半度以上才更新,就可以在温度差距达到半度时才调用setChanged(),进行有效的更新。其内幕如下:
[img]http://dl.iteye.com/upload/attachment/0063/5705/4967b30d-a0a6-30f4-86d4-c68f92a5d2b3.jpg[/img]

以上所有便是观察者模式啦~~
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值