OpenJDK源码研究笔记(六)--观察者模式工具类(Observer和Observable)和应用示例
本文主要讲解OpenJDK观察者模式的2个工具类,java.util.Observer观察者接口,java.util.Observable被观察者基类。
然后,给出了一个常见的观察者应用示例。
Observer观察者接口
/**
* 一个类可以实现Observer接口,当它想要得到“被观察者”对象变化通知的时候 */
public interface Observer {
/**
* 当被观察的对象发生变化时,这个方法被调用。一个应用调用Observable对象的notifyObservers方法,去通知所有的观察者, Observable对象发生了变化。 * @param o 被观察的对象.
* @param arg 传递给notifyObservers方法的参数.
*/
void update(Observable o, Object arg);
}
Observable被观察者基类
这个类代表一个可以观察的对象,或数据(在模型-视图范式中)。
它能够被继承去代表一个应用想要观察的对象。
一个observable对象可以有一个或多个观察者。
一个观察者可以是一个实现了Observer接口的对象。
当一个observable实例发生了变化,一个应用可以调用Observable的notifyObservers方法,去通知
它的观察者所发生的变化,通过调用观察者的update方法。
通知被传递的顺序是没有指定的。
在Observable类中的默认实现是,按注册顺序通知Observers。
但是子类可以改变这种顺序,使用没有保证的顺序,在分开的线程中传递通知,或者确保他们的子类遵循他们所选择的这个顺序。
注意:这个通知机制和线程没有任何关系,与Object对象的wait和notify机制没有任何关系。
当一个observable被新创建的时候,它的observer集合是空的。
两个observer被认为是同一个,当且仅当他们的equals方法比较返回true的时候。
public class Observable { //该对象是否已经发生了变化,默认为false
private boolean changed = false; //Vector是线程安全的,ArrayList是非线程安全的
private Vector obs; //构造一个没有观察者的Observable对象 public Observable() {
obs = new Vector();
} //增加一个观察者 public synchronized void addObserver(Observer o) {
if (o == null)
throw new NullPointerException();
if (!obs.contains(o)) {
obs.addElement(o);
}
} //删除一个观察者对象 public synchronized void deleteObserver(Observer o) {
obs.removeElement(o);
} public void notifyObservers() {
notifyObservers(null);
} public void notifyObservers(Object arg) {
//一个临时的数组缓冲,用来作为当前观察者状态的快照
Object[] arrLocal; synchronized (this) {
//我们不想要Observer 回调任何代码,在保持它的Monitor的时候。 //从Vector中提取Observable 和存储Observer的状态的代码需要同步,但是通知观察者却不需要。 //最坏的潜在竞争性条件结果是: //1.一个新增加的观察者将会错误正在进行的通知 //2.一个最近被取消注册的观察者将会被错误的通知,当它不需要关心的时候
if (!changed)
return;
arrLocal = obs.toArray();
clearChanged();
} //通知观察者的代码,在同步块之外 for (int i = arrLocal.length-1; i>=0; i--)
((Observer)arrLocal[i]).update(this, arg);
} //删除所有的观察者 public synchronized void deleteObservers() {
obs.removeAllElements();
} //表明Observable对象已经发生了变化 protected synchronized void setChanged() {
changed = true;
} //表明这个对象已经不再变化,或者说明已经通知到所有的观察者最近的一次变化。
protected synchronized void clearChanged() {
changed = false;
} //测试该对象是否已经发生了变化。当且仅当setChanged方法比clearChanged方法调用的次数更多的时候,返回true。 public synchronized boolean hasChanged() {
return changed;
} // 返回Observable对象的观察者的数量.
public synchronized int countObservers() {
return obs.size();
} }
代码示例
书,被观察者(或者称之为订阅主题)
import java.util.Observable; //书,被观察者(或者称之为订阅主题)
public class Book extends Observable { // 书的ID
private Integer id;
// 书名
private String name;
// 当前价格
private Double price; // 当书的价格修改时,调用该方法
public void modifyPrice(Double newPrice) {
ChangeStatus status = null;
// 差值大于0.01,就认为价格放生了变化
if (Math.abs(newPrice - price) >= 0.01) {
setChanged();
status = new ChangeStatus();
status.setId(id);
status.setName(name);
status.setOldPrice(price);
status.setNewPrice(newPrice); }
// 通知客户书已经降价
notifyObservers(status); } public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Double getPrice() {
return price;
} public void setPrice(Double price) {
this.price = price;
} public Integer getId() {
return id;
} public void setId(Integer id) {
this.id = id;
} }
观察者:用户的邮箱
import java.util.Observable;
import java.util.Observer; /**
* 观察者:用户的邮箱
*/
public class BuyerEmail implements Observer { // 该方法会被“被观察者的父类”即Observable调用
public void update(Observable o, Object arg) {
// 这里做具体的发电子邮件的程序
if (arg instanceof ChangeStatus) {
NotifyUtils.sendEmail((ChangeStatus) arg);
}
}
}
观察者:用户的手机
import java.util.Observable;
import java.util.Observer;
/**
* 观察者:用户的手机
*/
public class BuyerMobile implements Observer { // 该方法会被“被观察者的父类”即Observable调用
public void update(Observable o, Object arg) {
// 这里做具体的发电子邮件的程序
if(arg instanceof ChangeStatus){
NotifyUtils.sendSMS((ChangeStatus)arg);
}
} }
一本书的信息变化的实体类
public class ChangeStatus {
//书的ID
private Integer id;
//书名
private String name;
//过去的价格
private Double oldPrice;
//最新的价格
private Double newPrice; public Integer getId() {
return id;
} public void setId(Integer id) {
this.id = id;
} public Double getOldPrice() {
return oldPrice;
} public void setOldPrice(Double oldPrice) {
this.oldPrice = oldPrice;
} public Double getNewPrice() {
return newPrice;
} public void setNewPrice(Double newPrice) {
this.newPrice = newPrice;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} }
发送通知的工具类
public class NotifyUtils {
// 发送邮件通知用户
public static void sendEmail(ChangeStatus status) {
print(status);
} // 发送短信通知用户
public static void sendSMS(ChangeStatus status) {
print(status);
} // 一种只是用来作为“演示”的发送信息的方法
private static void print(ChangeStatus status) {
println("*******************************************************");
String messageContent = messageContent(status);
println(messageContent);
println("********************************************************");
} private static String messageContent(ChangeStatus status) {
// 一般,降价才会通知,这里只当是降价了
String str = "尊敬的用户,您好!\n";
str += "您收藏的图书《" + status.getName() + "》降价了。\n";
str += "过去的价格是:" + status.getOldPrice() + "\n";
str += "现在的价格是:" + status.getNewPrice() + "\n";
str += "您可以通过以下链接 http://blog.csdn.net/FansUnion 进行购买。";
return str;
} private static void println(Object content) {
System.out.println(content);
}
}
程序入口
public class ObserverPatternTest { /**
* 观察者模式-用法示例
*/
public static void main(String[] args) {
Book book = new Book();
book.setName("中国象棋程序的设计与实现");
book.setPrice(45.0); // 下面的观察者在实际应用中可以从数据库或文件中读取
BuyerEmail email = new BuyerEmail();
BuyerMobile mobile = new BuyerMobile(); // 增加观察者,在实际应用中就是哪些人对该书作了关注
book.addObserver(email);
book.addObserver(mobile); book.modifyPrice(34.00); }
}
输出结果
*******************************************************
尊敬的用户,您好!
您收藏的图书《中国象棋程序的设计与实现》降价了。
过去的价格是:45.0
现在的价格是:34.0
您可以通过以下链接 http://blog.csdn.net/FansUnion 进行购买。
********************************************************
*******************************************************
尊敬的用户,您好!
您收藏的图书《中国象棋程序的设计与实现》降价了。
过去的价格是:45.0
现在的价格是:34.0
您可以通过以下链接 http://blog.csdn.net/FansUnion 进行购买。
********************************************************
参考资料:OpenJDK7源码
原文参见: http://FansUnion.cn/articles/2936
OpenJDK源码研究笔记(六)--观察者模式工具类(Observer和Observable)和应用示例的更多相关文章
- OpenJDK源码研究笔记(八)-详细解析如何读取Java字节码文件(.class)
在上一篇OpenJDK源码研究笔记(七)–Java字节码文件(.class)的结构中,我们大致了解了Java字节码文件的结构. 本篇详细地介绍了如何读取.class文件的大部分细节. 1.构造文件 ...
- OpenJDK源码研究笔记(四)-编写和组织可复用的工具类和方法
本篇主要讲解java.util.Arrays这个针对数组的工具类. 1.可复用的工具类和方法. 这个工具类里,包含很多针对数组的工具方法,如 排序.交换.二分查找.比较.填充.复制.hashcode ...
- OpenJDK源码研究笔记(十四):三种经典的设计方法,接口,接口-抽象类-具体实现类,接口-具体实现类
在研究OpenJDK源码过程中,我发现常用的设计方法就是2种:接口,接口-抽象类-具体实现类 . 在一些其它开源框架和业务开发中,经常存在着第3种设计,接口-具体实现类. 1.只有接口,没有实现类. ...
- OpenJDK源码研究笔记(五)-缓存Integer等类型的频繁使用的数据和对象,大幅度提升性能(一道经典的Java笔试题)
摘要 本文先给出一个看似很简单实则有深意的Java笔试面试题,引出JDK内部的缓存. JDK内部的缓存,主要是为了提高Java程序的性能. 你能答对这道"看似简单,实则有深意"的J ...
- OpenJDK源码研究笔记(二)-Comparable和Comparator2个接口的作用和区别(一道经典的Java笔试面试题)
Comparable和Comparator是JDK中定义的2个比较接口,很相似,但又有所不同. 这2个接口的作用和区别也是Java中的常见经典面试题. 下面我们就来详细介绍下这2个接口的定义.作用.区 ...
- OpenJDK源码研究笔记(七)–Java字节码文件(.class)的结构
最近在看OpenJDK源码的过程中,顺便看了Java编译器(javac)的源码. 为了理解javac的源码,需要先搞懂Java字节码文件(.class)的结构. 于是,我就认真看了下OpenJDK中J ...
- OpenJDK源码研究笔记(十):枚举的高级用法,枚举实现接口,竟是别有洞天
在研究OpenJDK,Java编译器javac源码的过程中,发现以下代码. 顿时发现枚举类竟然也有如此"高端大气上档次"的用法. 沙场点兵(用法源码) com.sun.tools. ...
- OpenJDK源码研究笔记(十五):吐槽JDK中的10个富有争议的设计
前14篇文章,分享了JDK中值得学习和借鉴的编码和设计方法. 每个硬币都是有两面的.Every coin has two sides. 当然,JDK中也有很多值得改进或者说富有争议的设计. 本篇,就来 ...
- OpenJDK源码研究笔记(十三):Javac编译过程中的上下文容器(Context)、单例(Singleton)和延迟创建(LazyCreation)3种模式
在阅读Javac源码的过程中,发现一个上下文对象Context. 这个对象用来确保一次编译过程中的用到的类都只有一个实例,即实现我们经常提到的"单例模式". 今天,特意对这个上下文 ...
随机推荐
- 你不知道的JavaScript(十)with关键字
with关键字在JavaScript中不太常用,用来定义一个和对象相关的作用域,在该作用域中可以访问对象的属性或方法而前面无需加上对象名,以达到简化代码的目的. <script type=&qu ...
- UVa 1638 Pole Arrangement【递推】
题意:给出n根高度为1,2,3,---n的杆子,从左边能看到l根,右边能够看到r根,问有多少种可能 看的紫书的思路 先假设已经安排好了高度为2---i的杆子, 那么高度为1的杆子的放置方法有三种情况 ...
- HDU 2120 Ice_cream's world I【并查集】
解题思路:给出n对点的关系,求构成多少个环,如果对于点x和点y,它们本身就有一堵墙,即为它们本身就相连,如果find(x)=find(y),说明它们的根节点相同,它们之间肯定有直接或间接的相连,即形成 ...
- ActiveMQ学习笔记(14)----Destination高级特性(二)
1. Visual Destinations 1.1 概述 虚拟Destination用来创建逻辑Destinations,客户端可以通过它来产生和消费消息,它会把消息映射到物理Destination ...
- POJ-1276 Cash Machine 多重背包 二进制优化
题目链接:https://cn.vjudge.net/problem/POJ-1276 题意 懒得写了自己去看好了,困了赶紧写完这个回宿舍睡觉,明早还要考试. 思路 多重背包的二进制优化. 思路是将n ...
- luogu P2117 小Z的矩阵(结论题)
题意 题解 这题有点水. 我们发现对答案有贡献的实际上只有左上到右下的对角线上的数. 因为不在这条对角线上的乘积都要计算两遍,然后%2就都没了... 然后就做完了. #include<iostr ...
- [USACO10FEB]吃巧克力Chocolate Eating
题目:洛谷P2985. 题目大意:有n块巧克力要吃d天,并且只能按顺序吃.一块巧克力有一个开心值,吃了就能增加开心值.一个人初始开心值为0,且每天早上开心值变为原来的一半.问如何吃巧克力才能使开心值最 ...
- [Vijos P1369]难解的问题
题目大意:给你一个序列,叫你求最长上升子序列长度,但必须包含第k项. 解题思路:我们把k左边的比a[k]大的数去掉,k右边的比k小的数去掉,就可以保证选到a[k]了(因为左边的数小于a[k],而a[k ...
- LOJ #10121 与众不同 (RMQ+二分)
题目大意 :给你一个整数序列,定义一个合法子串为子串内所有数互不相同,会有很多询问,求区间$[L,R]$内最长连续合法子串长度 一道思维不错的$RMQ$题,NOIP要是考这种题可能会考挂一片 预处理出 ...
- mac pro配置php开发环境
mac pro自带php和apache,所以我们只要配置下就好了 // 启动Apache服务 sudo apachectl start // 重启Apache服务 sudo apachectl res ...