播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
视频:OB模式:观察者模式试用体验报告
发送到手机 / 微信 / 朋友圈
请扫描下列二维码
嵌入代码:
*通用代码支持手机播放哦
方式一:扫一扫
支持各类二维码扫描软件
方式二:发一发
免费发送App到手机
请输入正确的手机号码
看不清验证码不正确
该短信不收取任何费用
方式三:下一下
下载App观看
还有更多攻略和游戏礼包等着你
游戏新鲜报
&&英雄联盟
测试状态:(公测)
英雄联盟本周上升视频
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
播放: 评论:
畅游视频网页游戏YOYO手游
完美游戏台
网络视听许可证设计模式之观察者模式 - 简书
设计模式之观察者模式
图片来自网络
转载请注明出处:文章中的例子和思路均来自于《Head First》
我们接到一个来自气象局的需求:气象局需要我们构建一套系统,这系统有两个公告牌,分别用于显示当前的实时天气和未来几天的天气预报。当气象局发布新的天气数据(WeatherData)后,两个公告牌上显示的天气数据必须实时更新。气象局同时要求我们保证程序拥有足够的可扩展性,因为后期随时可能要新增新的公告牌。
这套系统中主要包括三个部分:气象站(获取天气数据的物理设备)、WeatherData(追踪来自气象站的数据,并更新公告牌)、公告牌(用于展示天气数据)
WeatherStation
WeatherData知道如何跟气象站联系,以获得天气数据。当天气数据有更新时,WeatherData会更新两个公告牌用于展示新的天气数据。
我们先来看看隔壁老王的实现:
public class WeatherData {
//实例变量声明
public void measurementsChanged() {
float temperature = getTemperature();
float humidity = getHumidity();
float pressure = getPressure();
List&Float& forecastTemperatures = getForecastTemperatures();
//更新公告牌
currentConditionsDisplay.update(temperature, humidity, pressure);
forecastDisplay.update(forecastTemperatures);
上面这段代码是典型的针对实现编程,这会导致我们以后增加或删除公告牌时必须修改程序。我们现在来看看观察者模式,然后再回来看看如何将观察者模式应用到这个程序。
观察者模式介绍
观察者模式面向的需求是:A对象(观察者)对B对象(被观察者)的某种变化高度敏感,需要在B变化的一瞬间做出反应。举个例子,新闻里喜闻乐见的***抓小偷,***需要在小偷伸手作案的时候实施抓捕。在这个例子里,***是观察者、小偷是被观察者,***需要时刻盯着小偷的一举一动,才能保证不会错过任何瞬间。程序里的观察者和这种真正的【观察】略有不同,观察者不需要时刻盯着被观察者(例如A不需要每隔1ms就检查一次B的状态),二是采用注册(_Register_)或者成为订阅(_Subscribe_)的方式告诉被观察者:我需要你的某某状态,你要在它变化时通知我。采取这样被动的观察方式,既省去了反复检索状态的资源消耗,也能够得到最高的反馈速度。
观察者模式通常基于Subject和__Observer__接口类来设计,下面是是类图:
观察者模式的应用
结合上面的类图,我们现在将观察者模式应用到WeatherData项目中来。于是有了下面这张类图:
ObserverForWeatherStation
* 主题(发布者、被观察者)
public interface Subject {
* 注册观察者
void registerObserver(Observer observer);
* 移除观察者
void removeObserver(Observer observer);
* 通知观察者
void notifyObservers();
观察者接口
public interface Observer {
void update();
公告牌用于显示的公共接口
public interface DisplayElement {
void display();
下面我们再来看看WeatherData是如何实现的
public class WeatherData implements Subject {
private List&Observer&
priva//温度
private List&Float& forecastT//未来几天的温度
public WeatherData() {
this.observers = new ArrayList&Observer&();
public void registerObserver(Observer observer) {
this.observers.add(observer);
public void removeObserver(Observer observer) {
this.observers.remove(observer);
public void notifyObservers() {
for (Observer observer : observers) {
observer.update();
public void measurementsChanged() {
notifyObservers();
public void setMeasurements(float temperature, float humidity,
float pressure, List&Float& forecastTemperatures) {
this.temperature =
this.humidity =
this.pressure =
this.forecastTemperatures = forecastT
measurementsChanged();
public float getTemperature() {
public float getHumidity() {
public float getPressure() {
public List&Float& getForecastTemperatures() {
return forecastT
显示当前天气的公告牌CurrentConditionsDisplay
public class CurrentConditionsDisplay implements Observer, DisplayElement {
private WeatherData weatherD
priva//温度
public CurrentConditionsDisplay(WeatherData weatherData) {
this.weatherData = weatherD
this.weatherData.registerObserver(this);
public void display() {
System.out.println("当前温度为:" + this.temperature + "℃");
System.out.println("当前湿度为:" + this.humidity);
System.out.println("当前气压为:" + this.pressure);
public void update() {
this.temperature = this.weatherData.getTemperature();
this.humidity = this.weatherData.getHumidity();
this.pressure = this.weatherData.getPressure();
display();
显示未来几天天气的公告牌ForecastDisplay
public class ForecastDisplay implements Observer, DisplayElement {
private WeatherData weatherD
private List&Float& forecastT//未来几天的温度
public ForecastDisplay(WeatherData weatherData) {
this.weatherData = weatherD
this.weatherData.registerObserver(this);
public void display() {
System.out.println("未来几天的气温");
int count = forecastTemperatures.size();
for (int i = 0; i & i++) {
System.out.println("第" + i + "天:" + forecastTemperatures.get(i) + "℃");
public void update() {
this.forecastTemperatures = this.weatherData.getForecastTemperatures();
display();
到这里,我们整个气象局的WeatherData应用就改造完成了。两个公告牌CurrentConditionsDisplay和ForecastDisplay实现了Observer和DisplayElement接口,在他们的构造方法中会调用WeatherData的registerObserver方法将自己注册成观察者,这样被观察者WeatherData就会持有观察者的应用,并将它们保存到一个集合中。当被观察者`WeatherData状态发送变化时就会遍历这个集合,循环调用观察者公告牌更新数据的方法。后面如果我们需要增加或者删除公告牌就只需要新增或者删除实现了Observer和DisplayElement接口的公告牌就好了。
观察者模式将观察者和主题(被观察者)彻底解耦,主题只知道观察者实现了某一接口(也就是Observer接口)。并不需要观察者的具体类是谁、做了些什么或者其他任何细节。任何时候我们都可以增加新的观察者。因为主题唯一依赖的东西是一个实现了Observer接口的对象列表。
好了,我们测试下利用观察者模式重构后的程序:
public class ObserverPatternTest {
public static void main(String[] args) {
WeatherData weatherData = new WeatherData();
CurrentConditionsDisplay currentConditionsDisplay = new CurrentConditionsDisplay(weatherData);
ForecastDisplay forecastDisplay = new ForecastDisplay(weatherData);
List&Float& forecastTemperatures = new ArrayList&Float&();
forecastTemperatures.add(22f);
forecastTemperatures.add(-1f);
forecastTemperatures.add(9f);
forecastTemperatures.add(23f);
forecastTemperatures.add(27f);
forecastTemperatures.add(30f);
forecastTemperatures.add(10f);
weatherData.setMeasurements(22f, 0.8f, 1.2f, forecastTemperatures);
输出结果:
当前温度为:22.0℃
当前湿度为:0.8
当前气压为:1.2
未来几天的气温
第0天:22.0℃
第1天:-1.0℃
第2天:9.0℃
第3天:23.0℃
第4天:27.0℃
第5天:30.0℃
第6天:10.0℃
源码地址:
Developer @Anjuke Inc.
个人博客:
Email: baron.