1 天气预报项目需求,详细要求如下:
1) 气象站可以将天天丈量到的温度,湿度,气压等等以通告的形式公布出去(好比公布到自己的网站或第三方)。
2) 需要设计开放型 API,便于其他第三方也能接入气象站获取数据。
3) 提供温度、气压和湿度的接口
4) 丈量数据更新时,要能实时的通知给第三方
2 天气预报设计方案 1-通俗方案
WeatherData 类
传统的设计方案
代码实现
packagecom.lin.observer;/*** 显示当前天气情况(可以明白为气象站的网站)
* @Description:
*@authorLinZM
* @date 2021-2-7 12:49:27
*@versionV1.8*/
public classCurrentConditions {//温度,气压,湿度
private floattemperatrue;private floatpressure;private floathumidity;//更新天气情况
public void update(float temperature, float pressure, floathumidity) {this.temperatrue =temperature;this.pressure =pressure;this.humidity =humidity;
display();
}public voiddisplay(){
System.out.println("===Today's temperature: "+temperatrue+"===");
System.out.println("===Today's pressure: "+pressure+"===");
System.out.println("===Today's humidity: "+humidity+"===");
}
}
packagecom.lin.observer;/** 类是焦点 1. 包罗最新的天气情况信息 2. 含有 CurrentConditions 工具 3. 当数据有更新时,就自动的挪用
* CurrentConditions 工具 update 方式(含 display), 这样他们(接入方)就看到最新的信息*/
public classWeatherData {private floattemperatrue;private floatpressure;private floathumidity;privateCurrentConditions currentConditions;//加入新的第三方
publicWeatherData(CurrentConditions currentConditions) {this.currentConditions =currentConditions;
}public floatgetTemperature() {returntemperatrue;
}public floatgetPressure() {returnpressure;
}public floatgetHumidity() {returnhumidity;
}public voiddataChange() {//挪用 接入方的 update
currentConditions.update(getTemperature(), getPressure(), getHumidity());
}//当数据有更新时,就挪用 setData
public void setData(float temperature, float pressure, floathumidity) {this.temperatrue =temperature;this.pressure =pressure;this.humidity =humidity;//挪用 dataChange, 将最新的信息 推送给 接入方 currentConditions
dataChange();
}
}
packagecom.lin.observer;public classClient {public static voidmain(String[] args) {
CurrentConditions currentConditions= newCurrentConditions();
WeatherData weatherData= newWeatherData(currentConditions);
weatherData.setData(10, 20, 3);
}
}
问题剖析
1) 其他第三方接入气象站获取数据的问题
2) 无法在运行时动态的添加第三方 (新浪网站)
3) 违反 ocp 原则=>观察者模式
//在 WeatherData 中,当增添一个第三方,都需要建立一个对应的第三方的通告板工具,并加入到 dataChange, 不利于维护,也不是动态加入
public void dataChange() {
currentConditions.update(getTemperature(), getPressure(), getHumidity());
}
3 观察者模式原理
1) 观察者模式类似订牛奶营业
2) 奶站/气象局:Subject
3) 用户/第三方网站:Observer
Subject:挂号注册、移除和通知
1) registerObserver 注 册
2) removeObserver 移 除
3) notifyObservers() 通知所有的注册的用户,凭据差别需求,可以是更新数据,让用户来取,也可能是实行推送, 看详细需求定
Observer:吸收输入
观察者模式:工具之间多对一依赖的一种设计方案,被依赖的工具为 Subject,依赖的工具为 Observer,Subject
通知 Observer 转变,好比这里的奶站是 Subject,是 1 的一方。用户时 Observer,是多的一方。
4 观察者模式解决天气预报需求
类图说明
代码实现
packagecom.lin.observer.plus;public interfaceSubject {voidregisterObserver(Observer o);voidremoveObserver(Observer o);voidnotifyObserver();
}
packagecom.lin.observer.plus;importjava.util.ArrayList;/** 类是焦点 1. 包罗最新的天气情况信息 2. 含有 CurrentConditions 工具 3. 当数据有更新时,就自动的挪用
* CurrentConditions 工具 update 方式(含 display), 这样他们(接入方)就看到最新的信息*/
public class WeatherData implementsSubject{private floattemperatrue;private floatpressure;private floathumidity;private ArrayListobservers;//加入新的第三方
publicWeatherData() {
observers= new ArrayList();
}public floatgetTemperature() {returntemperatrue;
}public floatgetPressure() {returnpressure;
}public floatgetHumidity() {returnhumidity;
}public voiddataChange() {//挪用 接入方的 update//currentConditions.update(getTemperature(), getPressure(), getHumidity());
notifyObserver();
}//当数据有更新时,就挪用 setData
public void setData(float temperature, float pressure, floathumidity) {this.temperatrue =temperature;this.pressure =pressure;this.humidity =humidity;//挪用 dataChange, 将最新的信息 推送给 接入方 currentConditions
dataChange();
}
@Overridepublic voidregisterObserver(Observer o) {
observers.add(o);
}
@Overridepublic voidremoveObserver(Observer o) {if(observers.contains(o)) {
observers.remove(o);
}
}
@Overridepublic voidnotifyObserver() {for (int i = 0; i < observers.size(); i++) {
observers.get(i).update(temperatrue, pressure, humidity);
}
}
}
packagecom.lin.observer.plus;public interfaceObserver {void update(float temperatrue, float pressure, floathumidity);
}
packagecom.lin.observer.plus;public class CurrentConditions implementsObserver{//温度,气压,湿度
private floattemperatrue;private floatpressure;private floathumidity;//更新天气情况
public void update(float temperature, float pressure, floathumidity) {this.temperatrue =temperature;this.pressure =pressure;this.humidity =humidity;
display();
}public voiddisplay(){
System.out.println("===Today's temperature: "+temperatrue+"===");
System.out.println("===Today's pressure: "+pressure+"===");
System.out.println("===Today's humidity: "+humidity+"===");
}
}
packagecom.lin.observer.plus;public class BaiDu implementsObserver{//温度,气压,湿度
private floattemperatrue;private floatpressure;private floathumidity;//更新天气情况
public void update(float temperature, float pressure, floathumidity) {this.temperatrue =temperature;this.pressure =pressure;this.humidity =humidity;
display();
}public voiddisplay() {
System.out.println("===BaiDu's temperature: " + temperatrue + "===");
System.out.println("===BaiDu's pressure: " + pressure + "===");
System.out.println("===BaiDu's humidity: " + humidity + "===");
}
}
packagecom.lin.observer.plus;public classClient {public static voidmain(String[] args) {//建立一个WeatherData
WeatherData weatherData = newWeatherData();//建立观察者
CurrentConditions currentConditions = newCurrentConditions();
BaiDu baiDu= newBaiDu();//注册
weatherData.registerObserver(currentConditions);
weatherData.registerObserver(baiDu);//测试
System.out.println("通知各个注册的观察者:");
weatherData.setData(23, 12, -0.4f);
}
}
观察者模式的利益
1) 观察者模式设计后,会以聚集的方式来治理用户(Observer),包罗注册,移除和通知。
2) 这样,我们增添观察者(这里可以明白成一个新的通告板),就不需要去修改焦点类 WeatherData 不会修改代码, 遵守了 ocp 原则。
5 观察者模式在 Jdk 应用的源码剖析
1) Jdk 的 Observable 类就使用了观察者模式
2) 代码剖析+模式角色剖析
3) 模式角色剖析
Observable 的作用和职位等价于 我们前面讲过 Subject
Observable 是类,不是接口,类中已经实现了焦点的方式 ,即治理 Observer 的方式 add.. delete .. notify…
Observer 的作用和职位等价于我们前面讲过的 Observer, 有 update
Observable 和 Observer 的使用方式和前面讲过的一样,只是 Observable 是类,通过继续来实现观察者模式
仅供参考,有错误还请指出!
有什么想法,谈论区留言,相互指教指教。