正文

一、定义

观察者模式定义了对象之间的一对多依赖,这样一来,当一个对象改变状态时,它的所有依赖者都会收到通知并自动更新。

要点:

  • 观察者模式定义了对象之间一对多的关系。
  • 观察者模式让主题(可观察者)和观察者之间松耦合。
  • 主题对象管理某些数据,当主题内的数据改变时,会以某种形式通知观察者。
  • 观察者可以订阅(注册)主题,以便在主题数据改变时能收到更新。
  • 观察者如果不想收到主题的更新通知,可以随时取消订阅(注册)。

二、实现步骤

1、创建主题父类/接口

主题父类/接口主要提供了注册观察者、移除观察者、通知观察者三个方法。

/**
* 主题
*/
public class Subject { /**
* 观察者列表
*/
private ArrayList<Observer> observers; public Subject() {
observers = new ArrayList<>();
} /**
* 注册观察者
*/
public void registerObserver(Observer o) {
observers.add(o);
} /**
* 移除观察者
*/
public void removeObserver(Observer o) {
observers.remove(o);
} /**
* 通知所有观察者,并推送数据(也可以不推送数据,而是由观察者过来拉取数据)
*/
public void notifyObservers(Object data) {
for (Observer o : observers) {
o.update(data);
}
}
}

2、创建观察者接口

观察者接口主要提供了更新方法,以供主题通知观察者时调用。

/**
* 观察者接口
*/
public interface Observer { /**
* 根据主题推送的数据进行更新操作
*/
public void update(Object data);
}

3、创建具体的主题,并继承主题父类/实现主题接口

/**
* 主题A
*/
public class SubjectA extends Subject { /**
* 主题数据
*/
private String data; public String getData() {
return data;
} public void setData(String data) {
this.data = data;
// 数据发生变化时,通知观察者
notifyObservers(data);
}
}

4、创建具体的观察者,并实现观察者接口

通过观察者类的构造函数,注册成为主题的观察者。

(1)观察者 A

/**
* 观察者A
*/
public class ObserverImplA implements Observer { private Subject subject; public ObserverImplA(Subject subject) {
// 保存主题引用,以便后续取消注册
this.subject = subject;
// 注册观察者
subject.registerObserver(this);
} @Override
public void update(Object data) {
System.out.println("Observer A:" + data.toString());
}
}

(2)观察者 B

/**
* 观察者B
*/
public class ObserverImplB implements Observer { private Subject subject; public ObserverImplB(Subject subject) {
// 保存主题引用,以便后续取消注册
this.subject = subject;
// 注册观察者
subject.registerObserver(this);
} @Override
public void update(Object data) {
System.out.println("Observer B:" + data.toString());
}
}

5、使用主题和观察者对象

public class Test {

    public static void main(String[] args) {
// 主题
SubjectA subject = new SubjectA();
// 观察者A
ObserverImplA observerA = new ObserverImplA(subject);
// 观察者B
ObserverImplB observerB = new ObserverImplB(subject);
// 模拟主题数据变化
subject.setData("I'm Batman!!!");
subject.setData("Why so serious...");
}
}

三、举个栗子

1、背景

你的团队刚刚赢得一纸合约,负责建立 Weather-O-Rama 公司的下一代气象站——Internet 气象观测站。

该气象站建立在 WeatherData 对象上,由 WeatherData 对象负责追踪目前的天气状况(温度、湿度、气压)。并且具有三种布告板,分别显示目前的状况、气象统计以及简单的预报。当 WeatherData 对象获得最新的测量数据时,三种布告板必须实时更新。

并且,这是一个可扩展的气象站,Weather-O-Rama 气象站希望公布一组 API,好让其他开发人员可以写出自己的气象布告板,并插入此应用中。

2、实现

(1)创建主题父类

/**
* 主题
*/
public class Subject { /**
* 观察者列表
*/
private ArrayList<Observer> observers; public Subject() {
observers = new ArrayList<>();
} /**
* 注册观察者
*/
public void registerObserver(Observer o) {
observers.add(o);
} /**
* 移除观察者
*/
public void removeObserver(Observer o) {
observers.remove(o);
} /**
* 通知所有观察者,并推送数据
*/
public void notifyObservers(float temperature, float humidity, float pressure) {
for (Observer o : observers) {
o.update(temperature, humidity, pressure);
}
}
}

(2)创建观察者接口

/**
* 观察者接口
*/
public interface Observer { /**
* 更新观测值
*/
public void update(float temperature, float humidity, float pressure);
}

(3)创建气象数据类,并继承主题父类

/**
* 气象数据
*/
public class WeatherData extends Subject { /**
* 温度
*/
private float temperature;
/**
* 湿度
*/
private float humidity;
/**
* 气压
*/
private float pressure; public void measurementsChanged() {
// 观测值变化时,通知所有观察者
notifyObservers(temperature, humidity, pressure);
} /**
* 设置观测值(模拟观测值变化)
*/
public void setMeasurements(float temperature, float humidity, float pressure) {
this.temperature = temperature;
this.humidity = humidity;
this.pressure = pressure;
measurementsChanged();
}
}

(4)创建布告板,并实现观察者接口

/**
* 目前状态布告板
*/
public class CurrentConditionsDisplay implements Observer { private Subject weatherData;
private float temperature;
private float humidity; public CurrentConditionsDisplay(Subject weatherData) {
this.weatherData = weatherData;
// 注册观察者
weatherData.registerObserver(this);
} @Override
public void update(float temperature, float humidity, float pressure) {
this.temperature = temperature;
this.humidity = humidity;
display();
} public void display() {
System.out.println("Current conditions:" + temperature + "F degress and " + humidity + "% humidity");
}
}
/**
* 统计布告板
*/
public class StatisticsDisplay implements Observer { private Subject weatherData;
private ArrayList<Float> historyTemperatures; public StatisticsDisplay(Subject weatherData) {
this.weatherData = weatherData;
// 注册观察者
weatherData.registerObserver(this);
historyTemperatures = new ArrayList<>();
} @Override
public void update(float temperature, float humidity, float pressure) {
this.historyTemperatures.add(temperature);
display();
} public void display() {
if (historyTemperatures.isEmpty()) {
return;
}
Collections.sort(historyTemperatures);
float avgTemperature = 0;
float maxTemperature = historyTemperatures.get(historyTemperatures.size() - 1);
float minTemperature = historyTemperatures.get(0);
float totalTemperature = 0;
for (Float temperature : historyTemperatures) {
totalTemperature += temperature;
}
avgTemperature = totalTemperature / historyTemperatures.size();
System.out.println("Avg/Max/Min temperature:" + avgTemperature + "/" + maxTemperature + "/" + minTemperature);
}
}
/**
* 预测布告板
*/
public class ForecastDisplay implements Observer { private Subject weatherData;
private float temperature;
private float humidity;
private float pressure; public ForecastDisplay(Subject weatherData) {
this.weatherData = weatherData;
// 注册观察者
weatherData.registerObserver(this);
} @Override
public void update(float temperature, float humidity, float pressure) {
this.temperature = temperature;
this.humidity = humidity;
this.pressure = pressure;
display();
} public void display() {
System.out.println("Forecast:waiting for implementation...");
}
}

(5)测试

public class Test {

    public static void main(String[] args) {
// 气象数据
WeatherData weatherData = new WeatherData();
// 目前状态布告板
CurrentConditionsDisplay currentConditionsDisplay = new CurrentConditionsDisplay(weatherData);
// 统计布告板
StatisticsDisplay statisticsDisplay = new StatisticsDisplay(weatherData);
// 预测布告板
ForecastDisplay forecastDisplay = new ForecastDisplay(weatherData); // 模拟气象观测值变化
weatherData.setMeasurements(80, 65, 30.4F);
weatherData.setMeasurements(82, 70, 29.2F);
weatherData.setMeasurements(78, 90, 29.2F);
}
}

最新文章

  1. 在linux下Ant的环境配置
  2. LeetCode Pow(x, n) (水题)
  3. 翻译:Knockout 轻松上手 - 1 Knockout 是什么?
  4. DataGridView控件
  5. 怎么加 一个 hyperlink 到 e-mail template for CRM
  6. NodeJS学习笔记—2.AMD规范
  7. jsp之间url传值出现中文乱码
  8. c++ primer( 文本查询程序)
  9. BZOJ 3720: Gty的妹子树 [树上size分块]
  10. Error RZ3007: Targeted tag name cannot be null or whitespace
  11. JS事件(二)事件对象
  12. bat处理复制文件
  13. Unix分类学习
  14. jQuery 作业三个按钮
  15. Unreal发展史
  16. 《DSP using MATLAB》Problem 4.4
  17. SQL Server MERGE
  18. springMVC 简单应用
  19. python&#39;s decorator&amp;wrapper
  20. HDU 5869 Different GCD Subarray Query rmq+离线+数状数组

热门文章

  1. Java实现分割矩形
  2. Java实现硬币收集问题
  3. Java实现第十届蓝桥杯迷宫
  4. 信道估计(channel estimation)图解——从SISO到MIMO原理介绍
  5. 汇编指令mov、add、sub、jmp
  6. 2.3 sqlmap目录及结构
  7. SpringSceurity(4)---短信验证码功能实现
  8. POJ - 2184 Cow Exhibition 题解
  9. 内核与驱动文件的version magic匹配问题
  10. laravel模板使用