《Head First设计模式》批注系列(一)——观察者设计模式

  最近在读《Head First设计模式》一书,此系列会引用源书内容,但文章内容会更加直接,以及加入一些自己的理解。

  观察者模式(有时又被称为模型-视图(View)模式、源-收听者(Listener)模式或从属者模式)。在此种模式中,一个目标物件管理所有相依于它的观察者物件,并且在它本身的状态改变时主动发出通知。这通常透过呼叫各观察者所提供的方法来实现。此种模式通常被用来实现事件处理系统。

  引子

  我们的公司刚刚你拿下了一个中国气象局的一个招标,负责建立北京市的气象观测站。该气象站必须建立在中国气象局专利申请中的WeatherData对象上,有WeatherData对象负责追踪目前的天气状况,(由于本文只做例子,只追踪温度、湿度、气压)。气象站提出应用的要求:

  1、我们公司能建立一个应用,有三种布告板,可以显示实时天气状况、气象统计、以及简单的预报。

  2、当WeatherData对象获得最新的测量数据时,三种布告板必须实时更新。

  3、我们的气象站是可以扩展的,气象站希望可以公布一组API(接口),好让其他开发人员可以自己写出自己的气象公告板,并插入此应用

  初步设计

  1、首先根据气象站的需求,我们画一个示意图。

  

  WeatherData对象知道如何跟物理气象站联系,当数据跟新时,WeatherData对象从气象站获得数据,WeatherData对象随机更新三个布告板的显示:目前状况(温度、湿度、气压)、气象统计和天气预报,所以我们目前的主要任务是,利用WeatherData对象从气象站获取数据,并跟新三个布告板。

  在设计之前,我们要看一看气象站给我们WeatherData类(此处只做示意,代码后面会变得完整)

public class WeatherData {
    //下面的三个方法获取温度、湿度、以及压力
    public float getTemperature(){
        temperature=获取温度;
        return temperature;
    }
    public float getHumidity(){
        temperature=获取湿度;
        return humidity;
    }
    public float getPressure(){
        temperature=获取压力;
        return pressure;
    }
    //一旦气象测量更新、此方法就会被调用
    public void measurementsChanged(){
        
    }

  好了,到此为止我们已经知道了最终的设计要求以及气象站给我们的WeatherData类,下面我们就要开始设计我们的应用,根据我们知道的东西,我们可以先来一个不完整的初版,我们现在WeatherData中增加一些方法(不着急先写出方法的具体内容,只要写出一个方法名,记得它的功能即可)

public class WeatherData {    
//下面的三个方法获取温度、湿度、以及压力 public float getTemperature(){ temperature=获取温度; return temperature; } public float getHumidity(){ temperature=获取湿度; return humidity; } public float getPressure(){ temperature=获取压力; return pressure; } //一旦气象测量更新、此方法就会被调用 public void measurementsChanged(){ float temp = getTemperature(); float humidity = getHumidity(); float pressure = getPressure();

     currentConditionsDisplay.update(temp,humidity,pressure);
     statisticsDiaplay.updata(temp,humidity,pressure);
     forecastDisplay.update(temp,humidity,pressure); }

  这是我们第一次设计的程序,由于我们已经开始准备用设计模式了,所以我们可以看看我们目前的代码中是不是有还可以改进的地方,首先,我们看到了三个公告板都有update方法,那我们就可以把这个方法

抽取成一个接口,这样我们每次增加公告板时直接实现此接口就可以,其次,本次我们讨论的是观察者模式,那我们的引子就先说到这,我们开始看看什么是观察者设计模式,我们了解完观察者设计模式后,我们

再回到我们的引子,完成设计。

  认识观察者设计模式

  在生活中,订阅报纸就是就是一个观察者设计模式,报社的业务就是出版报纸,当你订阅了某家报社的报纸,只要报社有了新的报纸出版,他们就会给你送来,只要你是他们的订户,你就一定会收到报

纸,当你不想在看这家报纸时,你就可以取消订阅,他们就不会再送来新报纸。其实,出版社+订阅者就等于观察者模式,只是在这个模式里,出版社叫做“主题(Subject)”,订阅者叫做观察者"(Observer)”,如果

你还没有看懂,那我们来看下面这个图:

  

  图中主题对象管理者int数据,狗对象、猫对象、老鼠对象订阅了主题,一旦主题对象发生改变,就会给这些订阅了的对象发送数据,鸭子对象没有订阅主题,就不会接收到数据。

  有一天,鸭子对象突然对主题对象管理的数据起了兴趣,它就需要过去告诉主题,我对你的数据改变有兴趣,一有变化请告诉我,这就是“订阅”的过程,在这之后,鸭子对象就是正式

的观察者了,以后主题管理的数据再有改变就会告诉鸭子对象。

  又有一天,老鼠对象对主题说,你管理的数据我观察太久了,我不想再当你的观察者了,这时主题就删掉老鼠对象,每当数据再变化时,老鼠对象就不会接收到数据。

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

  下面是便是定义观察者设计模式的类图:

  

  观察者设计模式应用到我们的气象站中

  我们已经基本了解了观察者设计模式,那我们就可以回到我们的引子完成原来的设计

  初步思考

  通过我们在上面分析的观察者设计模式,我们发现这是模式是一对多的关系,每当一改变时,就会给多发送改变的数据,而在我们的气象站设计中,正是一对多的关系,一就是我们的WeatherData数据

多就是显示的布告板,而不同的布告板有相同和不同的方法,我们可以把相同的部分抽取成接口,实现代码的复用,那么下面就让我们参照上面观察者设计模式的类图画一下我们气象站的设计类图。(自己

先尝试一下哟)

  设计气象站

  

    实现气象站

    根据上面的类图,我们就可以完善我们的代码啦

    Subject类的接口及WeatherData实现

package com.frank.observe.subject;
import com.frank.observe.observer.Observer;

public interface Subject {
  //主题注册、删除、通知观察者方法
public void registerObserver(Observer o); public void removeObserver(Observer o); public void notifyObserver(); //信息改变时执行方法 public void measurementsChanged(); }

 

package com.frank.observe.subject;
import java.util.ArrayList;
import com.frank.observe.observer.Observer;

public class WeatherData implements Subject {
  //WeatherData实现了Subject方法
private ArrayList<Observer> observers;
  //包含温度、湿度、压力
private float temperature; private float humidity; private float pressure; //构造器创建观察者们的集合 public WeatherData(){ observers = new ArrayList<Observer>(); } //接口方法 @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(o); } } @Override public void notifyObserver() { for(int i = 0;i<observers.size();i++){ Observer observer = observers.get(i); observer.update(temperature, humidity, pressure); } } @Override public void measurementsChanged() { notifyObserver(); }   //WeatherData自己的set方法,一会测试时在test中改变数据源 public void setMeasurements(float temperature,float humidity, float pressure){ this.temperature=temperature; this.humidity=humidity; this.pressure=pressure; measurementsChanged(); } }

 

    Observer类的接口以及CurrentConditionDisplay实现

package com.frank.observe.observer;
//Observer接口中更新数据的方法
public interface Observer {
    public void update(float temp,float humidity,float pressure);
}
package com.frank.observe.observer;
import com.frank.observe.display.DisplayElement;
import com.frank.observe.subject.Subject;
import com.frank.observe.subject.WeatherData;
//显示板要实现Observer以及DisplayElement接口(及公共方法)
public class CurrentConditionDisplay implements Observer, DisplayElement {
    private float temprature;
    private float humidity;
  //将weatherData作为自己的成员变量
private Subject weatherData; //在带参构造器中将自己注册到weatherData中 public CurrentConditionDisplay(Subject weatherData){ this.weatherData=weatherData; weatherData.registerObserver(this); }   //接口的更新方法 @Override public void update(float temp, float humidity, float pressure) { this.temprature=temp; this.humidity=humidity;
     //接口的显示方法 display(); } @Override
public void display() { System.out.println("current conditions" +temprature+"F degrees and"+humidity+"% humidity"); } }

 

  DisplayElement接口

package com.frank.observe.display;

public interface DisplayElement {
    public void display();
}

  Test方法

package com.frank.observe.test;
import com.frank.observe.observer.CurrentConditionDisplay;
import com.frank.observe.subject.WeatherData;

public class Test {
    public static void main(String[] args) {
     //新建一个主题 WeatherData weatherData
= new WeatherData();
     //新建一个公告板并将自己注册到主题中 CurrentConditionDisplay currentDisplay
= new CurrentConditionDisplay(weatherData);
     //每当主题更新数据时发送给观察者,观察者显示在公告板中 weatherData.setMeasurements(
80, 60, 30.4f); weatherData.setMeasurements(82, 70, 29.2f); weatherData.setMeasurements(78, 90, 29.2f); } }

 

   结果

current conditions80.0F degrees and60.0% humidity
current conditions82.0F degrees and70.0% humidity
current conditions78.0F degrees and90.0% humidity

 

   总结

   要点:

OO基础:抽象

OO原则:封装变化、多用组合,少用继承、针对接口编程,不针对实现编程

OO模式:观察者设计模式----在对象之间定义一对多的依赖,这样一来,当一个对象改变状态,依赖它的对象都会受到通知,并自动更新(后面的系列会看到其代表人物--MVC)

   

 

转载于:https://www.cnblogs.com/FrancisLiu/p/8728070.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
【优质项目推荐】 1、项目代码均经过严格本地测试,运行OK,确保功能稳定后才上传平台。可放心下载并立即投入使用,若遇到任何使用问题,随时欢迎私信反馈与沟通,博主会第一时间回复。 2、项目适用于计算机相关专业(如计科、信息安全、数据科学、人工智能、通信、物联网、自动化、电子信息等)的在校学生、专业教师,或企业员工,小白入门等都适用。 3、该项目不仅具有很高的学习借鉴价值,对于初学者来说,也是入门进阶的绝佳选择;当然也可以直接用于 毕设、课设、期末大作业或项目初期立项演示等。 3、开放创新:如果您有一定基础,且热爱探索钻研,可以在此代码基础上二次开发,进行修改、扩展,创造出属于自己的独特应用。 欢迎下载使用优质资源!欢迎借鉴使用,并欢迎学习交流,共同探索编程的无穷魅力! 基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip 基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip 基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。
观察者模式是一种常见的设计模式,它定义了一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖它的对象都会得到通知并自动更新。在这个模式中,被观察者对象通常称为主题(Subject),而观察者对象通常称为观察者(Observer)。 下面我们就以一个简单的天气预报系统为例来介绍观察者模式的使用。 首先,我们需要定义一个主题接口(Subject),它包含了添加、删除和通知观察者的方法: ```java public interface Subject { public void registerObserver(Observer o); public void removeObserver(Observer o); public void notifyObservers(); } ``` 然后,我们需要定义一个观察者接口(Observer),它包含了更新数据的方法: ```java public interface Observer { public void update(float temp, float humidity, float pressure); } ``` 接下来,我们需要定义一个具体的主题类(WeatherData),它实现了主题接口,并包含了一个列表来存储观察者对象,以及当前的温度、湿度和气压等数据: ```java import java.util.ArrayList; public class WeatherData implements Subject { private ArrayList<Observer> observers; private float temperature; private float humidity; private float pressure; public WeatherData() { observers = new ArrayList<Observer>(); } public void registerObserver(Observer o) { observers.add(o); } public void removeObserver(Observer o) { int i = observers.indexOf(o); if (i >= 0) { observers.remove(i); } } public void notifyObservers() { for (int i = 0; i < observers.size(); i++) { Observer observer = (Observer)observers.get(i); observer.update(temperature, humidity, pressure); } } public void measurementsChanged() { notifyObservers(); } public void setMeasurements(float temperature, float humidity, float pressure) { this.temperature = temperature; this.humidity = humidity; this.pressure = pressure; measurementsChanged(); } // other WeatherData methods here } ``` 最后,我们需要定义一个具体的观察者类(CurrentConditionsDisplay),它实现了观察者接口,并在更新数据时打印出当前的温度、湿度和气压等信息: ```java public class CurrentConditionsDisplay implements Observer { private float temperature; private float humidity; private Subject weatherData; public CurrentConditionsDisplay(Subject weatherData) { this.weatherData = weatherData; weatherData.registerObserver(this); } 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 degrees and " + humidity + "% humidity"); } } ``` 现在,我们可以创建一个天气预报系统,它包含了一个主题对象和一个观察者对象,并通过调用主题对象的方法来更新数据和通知观察者: ```java public class WeatherStation { public static void main(String[] args) { WeatherData weatherData = new WeatherData(); CurrentConditionsDisplay currentDisplay = new CurrentConditionsDisplay(weatherData); weatherData.setMeasurements(80, 65, 30.4f); weatherData.setMeasurements(82, 70, 29.2f); weatherData.setMeasurements(78, 90, 29.2f); } } ``` 以上就是一个简单的观察者模式的例子,它可以让我们更好地理解和应用这个常见的设计模式

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值