设计模式之观察者模式
在C#中有按钮的事件,在java swing中也有同样的事件,还接触过Android,同样也有这样的事件。我们想象一下,这些都用什么共同的特点?以c#为例,我们双击按钮就会出现按 钮事件的执行方法。在java中,自己也要添加监听事件,就收一个实现了ActionListener借口的类、有没有发现一点规律?就是当你触发某个东西时,会执行某个方法(操作)。Button类会监听按钮按下的事件,这个事件发生后会执行其他类中的代码!
想象一个下我们现实生活中的例子,你可以订阅报纸,每当有新闻发生(我们可以简单的理解每一期就是一个事件发生),你就会收到新的报纸。报社跟你之间没有其他的任何其他关系。
现有一个模式,我们有气象台给的WeatherData数据,需要我们将之显示到用户的设备上(目前状况、气象统计、天气预报三个布告板)。这是不是类似于订阅报纸的过程,当气象站有新的气象数据时,它会通知我们及时更新布告板。
那么,我们现在有什么呢?
1、WeatherData类具有gettter方法,可以取得三个测量值。
2、当新的数据准备好时,通知显示装置的方法(measurementChanged)会被调用。
3、收到新的数据,显示装置应立刻更新显示。
4、系统必须可扩展,以后随时增加或删除新的布告板。
那现在有一种情况,
public void measurementChanged(){
currentConditionDisplay.update(temp, humidity, pressure);
.........
.........
}
get方法省略没写,只写了一个布告板的update方法。这虽然能实现需要的功能,但是想象一下,扩展性好吗?如果增加一个布告板,是不是还要在WeatherData类中改变代码?这明显是不合理的,每一个布告板都对应着一个这样的调用。回忆一下策略模式,这里明显不是面向接口编程,而是面向实现编程。想到解决的办法了吗?下面我们要用观察者模式来解决这个问题。
那现在,我们正式的引入观察者模式。严格的定义:对象之间一对多依赖,这样一来,当每一个对象改变状态时,他的依赖着都会就收通知并自动更新。我们看看图例:
观察者对象都订
阅了某个事件,当事件发生时,主题对象就通知观察这对象。图中鸭子对象没有订阅,所以得不到通知。这时候设计就清楚了,为了设计的程序通用(以后增加或者
减少观察者),我们需要面向接口编程。设计主题、观察者、显示(每个观察者都需要显示,不变的地方拿出来:见策略模式)为接口,WeatherData实
现主题接口,各个布告板实现观察者和显示接口。这样脉络就清楚了,各个接口,类包含的功能见下面的图。
这里只简单的画了两个观察者,还有其他的观察者,如:气象统计,还有开发人员自己看的布告板。好了,图里面就不画了。奥,对了有没有法相主题和观察者之间的松耦合呢?这是OO设计原则非常重要的一点。设计原则:为了交互对象之间的松耦合设计而努力。
下面看我们实现的观察者模式的代码:
Subject接口实现:
public interface Subject {
public void registerObserver(Observer o);
public void removeObserver(Observer o);
public void notifyObservers();
}
Observer接口的实现:
public interface Observer {
public void update(float temp, float humidity, float presser);
}
DisplayElement接口的实现:
public interface DisplayElement {
public void display();
}
上面三个都是接口,我们提倡面向接口编程,可以保证代码的可拓展性。下面我们来看主题对象WeatherData(继承Subject接口)的代码:
public class WeatherData implements Subject{
private ArrayList observers;
private float temperature;
private float humidity;
private float pressure;
public WeatherData(){
observers = new ArrayList();
}
@Override
public void registerObserver(Observer o){
observers.add(o);
}
@Override
public void removeObserver(Observer o){
int i = observers.indexOf(o);
if(i >= 0){
observers.remove(i);
}
}
@Override
public void notifyObservers(){
for(int i = 0; i < observers.size(); i++){
Observer observer = (Observer)observers.get(i);
observer.update(temperature, humidity, pressure);
}
}
public void measurementChanged(){
notifyObservers();
}
public void setMeasurements(float temperature, float humidity, float pressure){
this.temperature = temperature;
this.humidity = humidity;
this.pressure = pressure;
measurementChanged();
}
}
下面各个布告板实现Observer和DispayElement接口:
public class CurrentConditionsDisplay implements Observer, DisplayElement{
private float temperature;
private float humidity;
private Subject weatherData;
public CurrentConditionsDisplay(Subject weatherData) {
// TODO Auto-generated constructor stub
this.weatherData = weatherData;
weatherData.registerObserver(this);
}
@Override
public void display() {
// TODO Auto-generated method stub
System.out.println("Current conditions: " + temperature + "F degrees and " + humidity + "% humidity");
}
@Override
public void update(float temp, float humidity, float presser) {
// TODO Auto-generated method stub
this.temperature = temp;
this.humidity = humidity;
display();
}
}
public class StatisticsDisplay implements Observer, DisplayElement {
private float maxTemp = 0.0f;
private float minTemp = 200;
private float tempSum= 0.0f;
private int numReadings;
private WeatherData weatherData;
public StatisticsDisplay(WeatherData weatherData) {
this.weatherData = weatherData;
weatherData.registerObserver(this);
}
public void update(float temp, float humidity, float pressure) {
tempSum += temp;
numReadings++;
if (temp > maxTemp) {
maxTemp = temp;
}
if (temp < minTemp) {
minTemp = temp;
}
display();
}
public void display() {
System.out.println("Avg/Max/Min temperature = " + (tempSum / numReadings)
+ "/" + maxTemp + "/" + minTemp);
}
}
public class ForecastDisplay implements Observer, DisplayElement {
private float currentPressure = 29.92f;
private float lastPressure;
private WeatherData weatherData;
public ForecastDisplay(WeatherData weatherData) {
this.weatherData = weatherData;
weatherData.registerObserver(this);
}
public void update(float temp, float humidity, float pressure) {
lastPressure = currentPressure;
currentPressure = pressure;
display();
}
public void display() {
System.out.print("Forecast: ");
if (currentPressure > lastPressure) {
System.out.println("Improving weather on the way!");
} else if (currentPressure == lastPressure) {
System.out.println("More of the same");
} else if (currentPressure < lastPressure) {
System.out.println("Watch out for cooler, rainy weather");
}
}
}
public class HeatIndexDisplay implements Observer, DisplayElement {
float heatIndex = 0.0f;
private WeatherData weatherData;
public HeatIndexDisplay(WeatherData weatherData) {
this.weatherData = weatherData;
weatherData.registerObserver(this);
}
public void update(float t, float rh, float pressure) {
heatIndex = computeHeatIndex(t, rh);
display();
}
private float computeHeatIndex(float t, float rh) {
float index = (float)((16.923 + (0.185212 * t) + (5.37941 * rh) - (0.100254 * t * rh)
+ (0.00941695 * (t * t)) + (0.00728898 * (rh * rh))
+ (0.000345372 * (t * t * rh)) - (0.000814971 * (t * rh * rh)) +
(0.0000102102 * (t * t * rh * rh)) - (0.000038646 * (t * t * t)) + (0.0000291583 *
(rh * rh * rh)) + (0.00000142721 * (t * t * t * rh)) +
(0.000000197483 * (t * rh * rh * rh)) - (0.0000000218429 * (t * t * t * rh * rh)) +
0.000000000843296 * (t * t * rh * rh * rh)) -
(0.0000000000481975 * (t * t * t * rh * rh * rh)));
return index;
}
public void display() {
System.out.println("Heat index is " + heatIndex);
}
}
再来启动测试程序:
public class WeatherStationHeatIndex {
public static void main(String[] args) {
// TODO Auto-generated method stub
WeatherData weatherData = new WeatherData();
CurrentConditionsDisplay currentDisplay = new CurrentConditionsDisplay(weatherData);
StatisticsDisplay statisticsDisplay = new StatisticsDisplay(weatherData);
ForecastDisplay forecastDisplay = new ForecastDisplay(weatherData);
HeatIndexDisplay heatIndexDisplay = new HeatIndexDisplay(weatherData);
weatherData.setMeasurements(80, 65, 30.4f);
weatherData.setMeasurements(82, 70, 29.2f);
weatherData.setMeasurements(78, 90, 29.2f);
}
}
相信代码可以看的更清楚,上面有哪里表达不清楚的,看了代码相信没有什么问题了。
本文作为方便自己复习而写,当然我也尽了最大努力写清楚,以方便同时其他人也能看懂,在自己学习的同时最好能够帮助到别人。当然,由于水平不足,肯定有什么错误或者不足的地方,希望大家指正批评。