设计模式之观察者模式(Java)
- 创建型模式,共5种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式;
- 结构型模式,共7种:适配器模式、装饰器模式、代理模式、外观模式、桥连接模式、组合模式、享元模式;
- 行为型模式,共11种:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。
关于上面23中设计模式的大概介绍可以参考博客[2]。
观察者模型实例之烧开水
从一个烧开水的例子出发。现在我们有个热水器,我们用它来烧开水。在烧水的过程中,显示器和扬声器都一直监视水温。显示器将水温显示在显示屏上。另外,当水温超过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)的更多相关文章
- 设计模式01观察者模式(java)
先发代码,有空来写内容. observer1 import java.util.Observer; import java.util.Observable; //学生类(Student)继承Obser ...
- 折腾Java设计模式之观察者模式
观察者模式 Define a one-to-many dependency between objects where a state change in one object results in ...
- 设计模式之第18章-观察者模式(Java实现)
设计模式之第18章-观察者模式(Java实现) 话说曾小贤,也就是陈赫这些天有些火,那么这些明星最怕的,同样最喜欢的是什么呢?没错,就是狗仔队.英文的名字比较有意思,是paparazzo,这一说法据说 ...
- 23种设计模式全解析 (java版本)
转自:http://blog.csdn.net/longyulu/article/details/9159589 其中PHP常用的五种设计模式分别为:工厂模式,单例模式,观察者模式,策略模式,命令模式 ...
- java_设计模式_观察者模式_Observer Pattern(2016-07-27)
看了好几篇文章,最终还是觉得<Head First 设计模式>举得例子比较符合观察者模式. 观察者模式概述: 观察者模式有时被称作发布/订阅模式,它定义了一种一对多的依赖关系,让多个观察者 ...
- 设计模式之观察者模式(Observable与Observer)
设计模式之观察者模式(Observable与Observer) 好久没有写博客啦,之前看完了<设计模式之禅>也没有总结一下,现在回忆一下设计模式之观察者模式. 1.什么是观察者模式 简单情 ...
- Head First 设计模式之观察者模式(Observer Pattern)
前言: 这一节开始学习观察者模式,开始讲之前会先像第一节那样通过一个应用场景来引入该模式.具体场景为:气象站提供了一个WeatherData对象,该对象可以追踪获取天气的温度.气压.湿度信息,Weat ...
- 从设计模式说起JAVA I/O流
从设计模式说起JAVA I/O流 之前写过一篇I/O流的入门介绍,直到最近看了设计模式,顺带理下I/O流的设计思路. JAVA类库中的I/O类分成输入和输出两部分,通过继承,任何自InputStrea ...
- 乐在其中设计模式(C#) - 观察者模式(Observer Pattern)
原文:乐在其中设计模式(C#) - 观察者模式(Observer Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 观察者模式(Observer Pattern) 作者:weba ...
- 8.5 GOF设计模式四: 观察者模式Observer
GOF设计模式四: 观察者模式Observer 现实中遇到的问题 当有许多不同的客户都对同一数据源感兴趣,对相同的数据有不同的处理方式,该如 何解决?5.1 定义: 观察者模式 观察者模式 ...
随机推荐
- c# 面相对象3-之继承性
继承:类与类之间的关系(父子关系) 子类继承父类,那么子类就拥有父类的公共属性和方法一个子类只能继承一个父类,一个父类可以有好多子类子类对象可以直接转成父类,但父类只能转回对应的子类如果子类转成父类之 ...
- sql server的两个类型转换函数
今天遇到一个sql的问题,条件中有个去当前月第一天(2013-8-23 0:00:00),很简单CAST(DATEADD(dd,-DAY(GETDATE())+1,GETDATE()) AS DATE ...
- 新浪微博iOS示例,登录,获取个人信息
1.导入第三方库和系统框架
- codeforces #330 div2
A: #include<cstdio> #include<algorithm> #include<cmath> #include<map> #inclu ...
- 网易云数据结构- Maximum Subsequence Sum
题目 题目地址 思路 显然是最大子列和的进化版,那就先思考下经典的最大子列和.这也是道思维题,啥算法也没用到,全是思维技巧,真心不知道考试遇到这种题该怎么办了. 存放答案的一个类,我把它看成一个袋子, ...
- 关于jQuery中的attr和data问题
今天在使用data获取属性并且赋值时遇到一个小问题,写下来防止以后再跳坑. 在使用jQuery获取自定义属性值时,我们习惯用 $(selector).attr('data-value'); jQuer ...
- SurfaceView的补充
1.什么时候使用:当自定义View需要频繁刷新,或者刷新数据比较大的时候,建议使用SurfaceView取代使用View 2.继承SurfaceView的并重写的步骤:①.继承SurfaceView类 ...
- php后台判断ajax请求
if (isset($_SERVER["HTTP_X_REQUESTED_WITH"]) && $_SERVER["HTTP_X_REQUESTED_WI ...
- 一个Java项目的学习
1. java命令行的启动 首先是gradle build 其次是:java -Dabc.appid=1234 -classpath "a.jar:b.jar" com.ctri ...
- SSD和HDD的区别
SSD与HDD最大的不同是:它没有马达.盘片.磁头摇臂这些HDD必需的机械部件,这是由两种硬盘不同的工作原理所决定的.SSD相比HDD来说节省了机械部件运动的时间,并且SSD所使用的主要存储元件NAN ...