设计模式(Design pattern)是什么?它是一套由四人组(The Gang of Four, [1]的作者)总结出来的软件设计框架。其目的是为了提高代码的可重用性,增强系统的可维护性和代码的易读性。在四人组的《Design Patterns》一书中提到了23中最常用的设计模式,大致可以分为三大类:
  1. 创建型模式,共5种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式;
  2. 结构型模式,共7种:适配器模式、装饰器模式、代理模式、外观模式、桥连接模式、组合模式、享元模式;
  3. 行为型模式,共11种:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。

  关于上面23中设计模式的大概介绍可以参考博客[2]。

  从上面的分类可以看出观察者模式是一种对象的行为型模式。它也被称之为发布-订阅(Publish-Subscribe)模式、模型视图(Model-View)模式或源-监听器(Source-Listener)模式[3]。通俗的说,观察者模式涉及到(至少)两个对象。其一是观察者,其二是被观察者。例如在一个RSS订阅系统中,发布方充当被观察者角色,而订阅方则充当了观察者角色。一般来说,观察者与被观察者是多对一的关系。即多个监听器可以同时监听某一主题对象。当主题对象状态发生改变时,监听器对象也会自动的更新自身状态。
 
  这篇博客主要是从一个实际的例子出发,来简单的说明Java(Android)、Objective-C(iOS)、C#(Windows Phone)三种面向对象编程语言对观察者模式的实现。
 

观察者模型实例之烧开水

  从一个烧开水的例子出发。现在我们有个热水器,我们用它来烧开水。在烧水的过程中,显示器和扬声器都一直监视水温。显示器将水温显示在显示屏上。另外,当水温超过90度时,扬声器开始发出警告,提示水快烧开。

Java实现

  Java对观察者类的支持,主要体现在Observable类和Observer接口。

1. java.util.Observable类 = 被观察者

Observable类的主要方法:

public class Observable {
void setChanged(); //设置被观察者的状态已经被改变
void clearChanged(); //清除被观察者状态的改变,此时再调用hasChanged()将返回false
void addObserver(Observer observer);
int countObservers();
void deleteObserver(Observer observer);
void deleteObservers();
boolean hasChanged();
void notifyObservers();
void notifyObservers(Object arg); // 参数一般设定为被改变的属性
}

热水器类继承自Observerable类,作为一种具体的被监控对象:

public class Heater extends Observable {
private int temperature; // get方法,返回水温
public int getTemperature(){
return temperature;
} // set方法,设置水温
public void setTemperature(int temperature){
this.temperature = temperature;
} // boilWater方法,烧水
public void boilWater(){
for(int i = 30; i <= 100; i++){
this.setTemperature(i);
this.setChanged();
this.notifyObservers(this.getTemperature());
this.clearChanged();
}
}
}

2. java.util.Observer接口 = 观察者

  Observer接口只有一个抽象方法需要被具体的观察者实现。

public interface Observer{
void update(Observable observable, Object arg); //这里的arg即Obserbale类里的notifyObservers(Object arg)传过来的参数
}

  Note:当被观察者调用nofityObservers(*)方法时,会根据被观察者的hasChanged()方法来判断它的状态是否被改变。如果改变,则观察者调用update方法。否则不调用(即不更新)。

  显示器和扬声器作为具体的热水器水温监视者,都需要实现Observer接口。对于显示器,它的update方法作用就是更新显示的水温。对扬声器,它的update方法则是在水温超过90度时发出警告。具体代码如下:

// 显示器
public class Displayer implements Observer {
public void update(Observable observable, Object arg){
display((int)arg);
}
  
public void display(int temperature){
System.out.println("Current Temperature is: " + temperature + " degree.");
}
} // 扬声器
public class Alertor implements Observer {
public void update(Observable observable, Object arg){
int temperature = (int) arg;
if(temperature > 90){
alarm();
}
} public void alarm(){
System.out.println("Warning: Temperature is over 90 degree!" );
}
}

3. 测试类

public class TestObserver{
public static void main(String[] args){
// 生成设备
Heater heater = new Heater();
Displayer displayer = new Displayer();
Alertor alertor = new Alertor(); // 添加订阅 - 重点(!!!)
heater.addObserver(display);
heater.addObserver(alertor); // 烧水
heater.boilWater();
}
}

4. 将上面的3个步骤统一起来,写到一个文件中如下:文件名TestObserver.java

import java.util.Observable;
import java.util.Observer; public class TestObserver {
public static void main(String[] args){
// 生成设备
Heater heater = new Heater();
Displayer displayer = new Displayer();
Alertor alertor = new Alertor(); // 添加订阅
heater.addObserver(displayer);
heater.addObserver(alertor); // 烧水
heater.boilWater();
}
} class Heater extends Observable {
private int temperature; // get方法,返回水温
public int getTemperature(){
return temperature;
} // set方法,设置水温
public void setTemperature(int temperature){
this.temperature = temperature;
} // boilWater方法,烧水
public void boilWater(){
for(int i = 30; i <= 100; i++){
this.setTemperature(i);
this.setChanged();
this.notifyObservers(this.getTemperature()); // 只需传入监视器感兴趣的变量
this.clearChanged();
}
}
} // 显示器
class Displayer implements Observer {
public void update(Observable observable, Object arg){
display((int)arg);
}
  
public void display(int temperature){
System.out.println("Displayer: Current Temperature is: " + temperature + " degree.");
}
} // 扬声器
class Alertor implements Observer {
public void update(Observable observable, Object arg){
int temperature = (int) arg;
if(temperature > 90){
alarm();
}
} public void alarm(){
System.out.println("Alertor: Warning: Temperature is over 90 degree!" );
}
}

最后的输出结果为:

Displayer: Current Temperature is:  30  degree.
Displayer: Current Temperature is: 31 degree.
Displayer: Current Temperature is: 32 degree.
...
Displayer: Current Temperature is: 90 degree.
Displayer: Current Temperature is: 91 degree.
Alertor: Warning: Temperature is over 90 degree!
Displayer: Current Temperature is: 92 degree.
Alertor: Warning: Temperature is over 90 degree!
...
Displayer: Current Temperature is: 99 degree.
Alertor: Warning: Temperature is over 90 degree!
Displayer: Current Temperature is: 100 degree.
Alertor: Warning: Temperature is over 90 degree!

总结

第一次写技术博客感觉比较费劲。原以为一个下午能把三种实现方法都写完的,结果只写完了Java部分。后续的Objective-C 和C#部分计划明天下午搞定。敬请期待。

欢迎交流和讨论^_^ ~ 如有任何问题,请发送到邮箱:fengfu0527@gmail.com

[1]. 《Design Patterns: Elements of Reusable Object-Oriented Software》

[2]. Java之美之设计模式,链接:http://bolg.csdn.net/zhangerqing/article/details/8194653

[3]. 《Java与模式》

 

设计模式之观察者模式(Java)的更多相关文章

  1. 设计模式01观察者模式(java)

    先发代码,有空来写内容. observer1 import java.util.Observer; import java.util.Observable; //学生类(Student)继承Obser ...

  2. 折腾Java设计模式之观察者模式

    观察者模式 Define a one-to-many dependency between objects where a state change in one object results in ...

  3. 设计模式之第18章-观察者模式(Java实现)

    设计模式之第18章-观察者模式(Java实现) 话说曾小贤,也就是陈赫这些天有些火,那么这些明星最怕的,同样最喜欢的是什么呢?没错,就是狗仔队.英文的名字比较有意思,是paparazzo,这一说法据说 ...

  4. 23种设计模式全解析 (java版本)

    转自:http://blog.csdn.net/longyulu/article/details/9159589 其中PHP常用的五种设计模式分别为:工厂模式,单例模式,观察者模式,策略模式,命令模式 ...

  5. java_设计模式_观察者模式_Observer Pattern(2016-07-27)

    看了好几篇文章,最终还是觉得<Head First 设计模式>举得例子比较符合观察者模式. 观察者模式概述: 观察者模式有时被称作发布/订阅模式,它定义了一种一对多的依赖关系,让多个观察者 ...

  6. 设计模式之观察者模式(Observable与Observer)

    设计模式之观察者模式(Observable与Observer) 好久没有写博客啦,之前看完了<设计模式之禅>也没有总结一下,现在回忆一下设计模式之观察者模式. 1.什么是观察者模式 简单情 ...

  7. Head First 设计模式之观察者模式(Observer Pattern)

    前言: 这一节开始学习观察者模式,开始讲之前会先像第一节那样通过一个应用场景来引入该模式.具体场景为:气象站提供了一个WeatherData对象,该对象可以追踪获取天气的温度.气压.湿度信息,Weat ...

  8. 从设计模式说起JAVA I/O流

    从设计模式说起JAVA I/O流 之前写过一篇I/O流的入门介绍,直到最近看了设计模式,顺带理下I/O流的设计思路. JAVA类库中的I/O类分成输入和输出两部分,通过继承,任何自InputStrea ...

  9. 乐在其中设计模式(C#) - 观察者模式(Observer Pattern)

    原文:乐在其中设计模式(C#) - 观察者模式(Observer Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 观察者模式(Observer Pattern) 作者:weba ...

  10. 8.5 GOF设计模式四: 观察者模式Observer

    GOF设计模式四: 观察者模式Observer  现实中遇到的问题  当有许多不同的客户都对同一数据源感兴趣,对相同的数据有不同的处理方式,该如 何解决?5.1 定义: 观察者模式  观察者模式 ...

随机推荐

  1. django学习笔记一

    django作为一个python的开源项目发布,其web框架采用了mtv设计模式 在目前一些较为成熟的大型网站中有不少网站的应用基于django开发,django作为一个重量型的web框架提供了以下的 ...

  2. WindowsForm 打印

    打印: 打印对话框:printdialog 页面设置:pagesetupdialog 这两个对话框都需要通过设置printdocument来指定打印对象 printdocument:打印对象,必须要有 ...

  3. Aandroid Error之 新导入工程报Unable to resolve target 'android-18'和R cannot be resolved

    有段时间没有写安卓了,今天导入以前的项目,结果看到控制台打印出了这样一句:Unable to resolve target 'android-18', 解决方法: 项目->属性->Andr ...

  4. [Effective Modern C++] Item 2. Understand auto type deduction - 了解auto类型推断

    条款二 了解auto类型推断 基础知识 除了一处例外,auto的类型推断与template一样.存在一个直接的从template类型推断到auto类型推断的映射 三类情况下的推断如下所示: // ca ...

  5. MySQL server has gone away报错

    1.最近做插入数据库,然后一直报一个错.mysql server has gone away.(如下图) 查了好多资料,终于解决了.. 1.可能是连接超时..进入php.ini,修改wait_time ...

  6. JSP作为img的src时chrome的bug

    废话不说,直接上代码. 新建一个jsp文件,名为img.jsp,代码如下: <%@page import="javax.imageio.ImageIO"%> <% ...

  7. tp可用的超强第三方图表类库-JpGraph

    日常开发中经常需要做图表,比如线状图.饼状图.柱状图等等,用PHP作图时需要使用复杂抽象的画图函数,或者借助一些网上下载的花柱形图.饼形图的类来实现,没有一个统一的chart类来实现图表的快速开发,非 ...

  8. cocos2d-x中的Tiled地图

    cocos2d-x中的瓦片地图是通过tiledMap软件制作的,存档格式是.tmx格式.此软件的使用步骤简单总结如下: (1)制作瓦片地图 1 打开软件,软件界面如下图. 2. 新建地图(文件-> ...

  9. 项目 Web 的 NuGet 程序包还原失败: 找不到“1.0.0”版本的程序包“Microsoft.Net.Compilers”。。 0

    项目   Web 的 NuGet 程序包还原失败: 找不到“1.0.0”版本的程序包“Microsoft.Net.Compilers”.. 0 使用vs的NutGet包管理器时,另一台电脑从svn下载 ...

  10. python核心编程-第五章-个人笔记

    1.用del删除对对象的引用 >>> a = 123 >>> a 123 >>> del a >>> a Traceback ( ...