《Head First 设计模式》[02] 观察者模式
1、观察者模式
1.1 形象地认识观察者模式
- 报社的业务是出版报纸
- 用户像某家报社订阅了报纸,那么一旦报社有新的报纸,就会送到用户处。只要是订户,就一直会收到新报纸;
- 当用户不再想看报纸时,取消订阅,报社则不再送新的报纸来


1.2 定义观察者模式

1.3 网络气象站的建立故事
1.3.1 故事背景
- 气象站会提供WeatherData对象,由其追踪目前的天气情况
- 外包公司要建立一个应用,有三种布告板,分别显示目前的状况、气象统计、简单预报
- 当WeatherData对象获取到新的数据时,三种布告板必须更新
- 要求可拓展,能让其他开发者根据API自定义公告板

1.3.2 目前有什么
- getTemperature()
- getHumidity()
- getPressure()
- 气象测量数据更新时,该方法会被调用(你不需要关心怎么调用的,这里只是条件,你只需要知道该方法会被调用即可)
- 该方法需要外包公司实现具体代码
1.3.3 要做什么
- 实现三个基本的布告板,且WeatherData有数据更新时,布告板的信息也必须更新
- 系统可扩展,让其他开发人员可以自定义布告板,且可随意添加和删除
1.3.4 错误的示范
public void measurementChanged() {
float temp = getTemperature();
float humidity = getHumidity();
float pressure = getPressure();
currentConditionsDisplay.update(temp, humidity, pressure);
statisticsDisplay.update(temp, humidity, pressure);
forecastDisplay.update(temp, humidity, pressure);
}
public void measurementChanged() {
float temp = getTemperature();
float humidity = getHumidity();
float pressure = getPressure();
currentConditionsDisplay.update(temp, humidity, pressure);
statisticsDisplay.update(temp, humidity, pressure);
forecastDisplay.update(temp, humidity, pressure);
}
1.3.5 观察者模式的威力
public interface Subject {
void registerObserver(Observer o);
void removeObserver(Observer o);
void notifyObservers();
}
public interface Subject {
void registerObserver(Observer o);
void removeObserver(Observer o);
void notifyObservers();
}
public interface Observer {
void update(float temp, float humidity, float pressure);
}
public interface Observer {
void update(float temp, float humidity, float pressure);
}
public interface DisplayElement {
void display();
}
public interface DisplayElement {
void display();
}
public class WeatherData implements Subject{
//新增订阅者集合属性
private ArrayList<Observer> observerList = new ArrayList<Observer>();
private float temperature;
private float humidity;
private float pressure;
public float getTemperature() {
return temperature;
}
public float getHumidity() {
return humidity;
}
public float getPressure() {
return pressure;
}
@Override
public void registerObserver(Observer o) {
if (!observerList.contains(o)) {
observerList.add(o);
}
}
@Override
public void removeObserver(Observer o) {
if (observerList.contains(o)) {
observerList.remove(o);
}
}
@Override
public void notifyObservers() {
for (Observer o : observerList) {
o.update(temperature, humidity, pressure);
}
}
public void measurementChanged() {
notifyObservers();
}
}
public class WeatherData implements Subject{
//新增订阅者集合属性
private ArrayList<Observer> observerList = new ArrayList<Observer>();
private float temperature;
private float humidity;
private float pressure;
public float getTemperature() {
return temperature;
}
public float getHumidity() {
return humidity;
}
public float getPressure() {
return pressure;
}
@Override
public void registerObserver(Observer o) {
if (!observerList.contains(o)) {
observerList.add(o);
}
}
@Override
public void removeObserver(Observer o) {
if (observerList.contains(o)) {
observerList.remove(o);
}
}
@Override
public void notifyObservers() {
for (Observer o : observerList) {
o.update(temperature, humidity, pressure);
}
}
public void measurementChanged() {
notifyObservers();
}
}
public class CurrentConditionsDisplay implements Observer, DisplayElement {
private float temperature;
private float humidity;
private float pressure;
private Subject weatherData;
public CurrentConditionsDisplay(Subject weatherData) {
//保留Subject的引用,将来取消注册时会比较方便
this.weatherData = weatherData;
weatherData.registerObserver(this);
}
@Override
public void update(float temp, float humidity, float pressure) {
//把温度、湿度、压力先保存一下,再调用display方法展示
this.temperature = temp;
this.humidity = humidity;
this.pressure = pressure;
display();
}
@Override
public void display() {
System.out.println("currentConditionsDisplay:");
System.out.println(" temperature:" + this.temperature);
System.out.println(" humidity:" + this.humidity);
System.out.println(" pressure:" + this.pressure);
}
}
public class CurrentConditionsDisplay implements Observer, DisplayElement {
private float temperature;
private float humidity;
private float pressure;
private Subject weatherData;
public CurrentConditionsDisplay(Subject weatherData) {
//保留Subject的引用,将来取消注册时会比较方便
this.weatherData = weatherData;
weatherData.registerObserver(this);
}
@Override
public void update(float temp, float humidity, float pressure) {
//把温度、湿度、压力先保存一下,再调用display方法展示
this.temperature = temp;
this.humidity = humidity;
this.pressure = pressure;
display();
}
@Override
public void display() {
System.out.println("currentConditionsDisplay:");
System.out.println(" temperature:" + this.temperature);
System.out.println(" humidity:" + this.humidity);
System.out.println(" pressure:" + this.pressure);
}
}
1.4 JDK内置的观察者模式 Observer和Observable
- Observable是类,进行了方法的拓展,不再是单纯的接口
- Observable包含setChange()方法,用来标记状态已经改变的事实,且notifyObservers()仅在change为true时通知
- 以更好自定义推送粒度
- 如温度控制很精确每一点点都在变化,会造成持续不断地通知
- 如果我们希望温度变化在1度以上才进行通知,那么就可以在温度变化达到1度时,调用setChange()
- notifyObservers()有重载
- notifyObservers()
- notifyObservers(Object arg)
- 如果需要推“push”数据,则将数据自定义封装为某个数据对象arg(即主动通知观察者,改变的数据是什么)
- 则Observer的update(Observable o, Object arg)可以直接使用arg
- 如果需要拉“pull”数据,则调用notifyObservers(),实际调用notifyObservers(null),观察者需要什么数据,自己去"拉pull"
- 则Observer的update(Observable o, Object arg)需要通过参数Observable的getter获取想要的信息
- 继承Observable,而不再是实现接口
- 不再需要“为了记住观察者们而新增观察者的集合属性”,因为Observable中已经有了
- 通知观察者前必须调用setChanged()方法
public class WeatherData extends Observable{
private float temperature;
private float humidity;
private float pressure;
public WeatherData() {
}
public float getTemperature() {
return temperature;
}
public float getHumidity() {
return humidity;
}
public float getPressure() {
return pressure;
}
public void measurementChanged() {
setChanged();
notifyObservers();
}
}
public class WeatherData extends Observable{
private float temperature;
private float humidity;
private float pressure;
public WeatherData() {
}
public float getTemperature() {
return temperature;
}
public float getHumidity() {
return humidity;
}
public float getPressure() {
return pressure;
}
public void measurementChanged() {
setChanged();
notifyObservers();
}
}
- 因为是拉pull,所以先确定被观察者的类型
public class CurrentConditionsDisplay implements Observer, DisplayElement {
private float temperature;
private float humidity;
private float pressure;
private Observable weatherData;
public CurrentConditionsDisplay(Observable weatherData) {
this.weatherData = weatherData;
weatherData.addObserver(this);
}
@Override
public void update(Observable o, Object arg) {
if (o instanceof WeatherData) {
WeatherData weatherData = (WeatherData) o;
this.temperature = weatherData.getTemperature();
this.humidity = weatherData.getHumidity();
this.pressure = weatherData.getPressure();
}
display();
}
@Override
public void display() {
System.out.println("currentConditionsDisplay:");
System.out.println(" temperature:" + this.temperature);
System.out.println(" humidity:" + this.humidity);
System.out.println(" pressure:" + this.pressure);
}
}
public class CurrentConditionsDisplay implements Observer, DisplayElement {
private float temperature;
private float humidity;
private float pressure;
private Observable weatherData;
public CurrentConditionsDisplay(Observable weatherData) {
this.weatherData = weatherData;
weatherData.addObserver(this);
}
@Override
public void update(Observable o, Object arg) {
if (o instanceof WeatherData) {
WeatherData weatherData = (WeatherData) o;
this.temperature = weatherData.getTemperature();
this.humidity = weatherData.getHumidity();
this.pressure = weatherData.getPressure();
}
display();
}
@Override
public void display() {
System.out.println("currentConditionsDisplay:");
System.out.println(" temperature:" + this.temperature);
System.out.println(" humidity:" + this.humidity);
System.out.println(" pressure:" + this.pressure);
}
}
- Observable是一个类而不是接口,而Java不能多继承,限制了它的使用
- setChanged()方法是protected修饰,所以除非你继承自Observable,否则你无法创建一个Observable实例并组合到你自己的对象中
2、再多叨叨两句
3、本文涉及的设计原则
- 为了交互对象之间的松耦合设计而努力
4、相关好文推荐
《Head First 设计模式》[02] 观察者模式的更多相关文章
- java设计模式02观察者模式
观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象.这个主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己. 这里主要讲一下学习内置观察者的记录,在JA ...
- php 设计模式之观察者模式(订阅者模式)
php 设计模式之观察者模式 实例 没用设计模式的代码,这样的代码要是把最上面那部分也要符合要求加进来,就要修改代码,不符合宁增不改的原则 介绍 观察者模式定义对象的一对多依赖,这样一来,当一个对象改 ...
- 乐在其中设计模式(C#) - 观察者模式(Observer Pattern)
原文:乐在其中设计模式(C#) - 观察者模式(Observer Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 观察者模式(Observer Pattern) 作者:weba ...
- 设计模式之观察者模式(Observable与Observer)
设计模式之观察者模式(Observable与Observer) 好久没有写博客啦,之前看完了<设计模式之禅>也没有总结一下,现在回忆一下设计模式之观察者模式. 1.什么是观察者模式 简单情 ...
- 8.5 GOF设计模式四: 观察者模式Observer
GOF设计模式四: 观察者模式Observer 现实中遇到的问题 当有许多不同的客户都对同一数据源感兴趣,对相同的数据有不同的处理方式,该如 何解决?5.1 定义: 观察者模式 观察者模式 ...
- iOS设计模式(02):单例模式
iOS设计模式(02):单例模式 singleton-design-pattern 什么是单例模式? 单例模式是一个类在系统中只有一个实例对象.通过全局的一个入口点对这个实例对象进行访问.在iOS开发 ...
- [JS设计模式]:观察者模式(即发布-订阅者模式)(4)
简介 观察者模式又叫发布---订阅模式,它定义了对象间的一种一对多的关系,让多个观察者对象同时监听某一个主题对象,当一个对象发生改变时,所有依赖于它的对象都将得到通知. 举一个现实生活中的例子,例如小 ...
- 实践GoF的23种设计模式:观察者模式
摘要:当你需要监听某个状态的变更,且在状态变更时通知到监听者,用观察者模式吧. 本文分享自华为云社区<[Go实现]实践GoF的23种设计模式:观察者模式>,作者: 元闰子 . 简介 现在有 ...
- OOP设计模式[JAVA]——02观察者模式
观察者模式 观察者模式的设计原则 为交互对象之间的松耦合设计而努力,使对象之间的相互依赖降到最低. 观察者模式也是对象行为型模式,其意图为:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时 ...
随机推荐
- JavaScript基础要点
一.值和类型及运算 JavaScript中的六种基本值类型 数字(number).字符串(string).布尔值(boolean).对象(object).函数(function).未定义类型(unde ...
- js识别设备
console.log(window.navigator); Navigator 对象属性 appCodeName 返回浏览器的代码名. appMinorVer ...
- python的四大函数讲解
Python的四类函数: 1.普通函数 2.默认函数 3.关键字函数 4.收集参数 1.普通函数 a.定义的时候直接定义变量名 b.调用的时候直接把变量或者值放入指定位置 def 函数名(参数1,参数 ...
- 2017-12-22 日语编程语言"抚子"-第三版实现初探
前文日语编程语言"抚子" - 第三版特色初探仅对语言的语法进行了初步了解. 之前的语言原型实现尝试(如编程语言试验之Antlr4+JavaScript实现"圈4" ...
- React报错:Can't perform a React state update on an unmounted component. This is a no-op, but it indicates a memory leak in your application. To fix,
今天在开发时报了以下错误,记录一下 我们不能在组件销毁后设置state,防止出现内存泄漏的情况 出现原因直接告诉你了,组件都被销毁了,还设置个锤子的state啊 解决方案: 利用生命周期钩子函数:co ...
- 长文本溢出显示省略号(…) text-overflow: ellipsis
text-overflow 属性规定当文本溢出包含元素时发生的事情. 默认值: clip 继承性: no 版本: CSS3 JavaScript 语法: object .style.textOverf ...
- JS性能优化 之 文档片段 createDocumentFragment
我们用原生JS进行开发时,经常会用到两种更新DOM节点的方法:innerHTML 和 appendChild() .其中 innerHTML 会完全替换掉原先的节点内容,如果我们是想向元素追加子节点的 ...
- python自动化开发-9 进程 线程
进程与线程 程序并不能单独运行,只有将程序装载到内存中,系统为它分配资源才能运行,而这种执行的程序就称之为进程.线程是操作系统能够进行运算调度的最小单位.它被包含在进程之中,是进程中的实际运作单位.一 ...
- linux定时任务调度定系统——opencron
linux定时任务调度定系统——opencron https://gitee.com/terrytan/opencron/#%E8%BF%90%E8%A1%8C%E7%8E%AF%E5%A2%83 一 ...
- SG Input 软件安全分析之逆向分析
前言 通过本文介绍怎么对一个 windows 程序进行安全分析.分析的软件版本为 2018-10-9 , 所有相关文件的链接 链接:https://pan.baidu.com/s/1l6BuuL-HP ...