设计模式(二)观察者模式

1.前言

本期分享设计模式主题:观察者设计模式。

这是一个非常简单易学的设计模式,读完本文,你能知道观察者模式的设计以及设计思想。

2.模式简介

这是一个可以帮助你的对象知悉现状,不会错过对该对象感兴趣的事。对象甚至在运行时可决定是否需要继续被通知。观察者模式也是JDK中使用最多的模式之一,非常有用。

3.模式定义

观察者模式是在对象之间,定义一对多的依赖关系。基于这样的关系,如果对象发生状态变化,与之依赖的多个对象(监听的对象)能够收到状态变化通知,并进行更新。是一个以松耦合的方式,让一个对象与一系列对象进行协作沟通的方式。

观察者模式有两种方式的实现,一种是推(push),另一种是主动拉(pull),推和拉都是针对通知而言。本着介绍推模式。

4.模式UML类图

观察者模式的UML类图如下图:


观察者模式UML类图

角色划分

  • Subject:主题或中心对象,一对多中的一,接口或抽象类。定义了如何添加观察者,移除观察者,通知观察者更新的基本行为。
  • SubjectImpl1和SujectImpl2:是对主题的具体实现,可以通过实现Subject的接口实现不同种类的主题和中心对象。
  • Observer:观察者接口定义。定义观察者更新行为。
  • ObserverImpl1和ObserverImpl2:观察者的实现类,可以通过实现Observer接口实现不同的观察者,监听不同的主题。

5.代码实例

实例介绍

实例实现了一个天气中心,向一个当前天气显示器上通知最新的气象数据:温度、湿度、压强。气象显示器会更新显示的数据,并实时显示。

主题中心类:

/**
 * 主题类
 * @author Misout
 * @date 2018-03-25 12:40:37
 */
public interface Subject {
    /**
     * 注册添加观察者
     * @param observer
     */
    void registerObserver(Observer observer);
    /**
     * 移除观察者
     * @param observer
     */
    void removeObserver(Observer observer);
    /**
     * 更新或通知变更所有观察者
     */
    void notifyObservers();
}

具体主题实现类:

/**
 * 具体主题实现类:一个天气数据中心
 * 一旦更新数据,实时通知观察者
 * @author Misout
 * @date 2018-03-25 12:51:19
 */
public class WeatherData implements Subject {
    /** 已注册的观察者列表 */
    private List<Observer> observers;
    /** 温度 */
    private float temp;
    /** 湿度 */
    private float humidity;
    /** 压强 */
    private float pressure;
    public WeatherData() {
        observers = new ArrayList<Observer>();
    }
    @Override
    public void registerObserver(Observer observer) {
        observers.add(observer);
    }
    @Override
    public void removeObserver(Observer observer) {
        int index = observers.indexOf(observer);
        if(index >= 0) {
            observers.remove(index);
        }
    }
    @Override
    public void notifyObservers() {
        for (Observer observer : observers) {
            observer.update(temp, humidity, pressure);
        }
    }
    /**
     * 发生数据变化时,实时通知观察者
     */
    public void measurementsChanged() {
        notifyObservers();
    }
    public void setMeasurements(float temp, float humidity, float pressure) {
        this.temp = temp;
        this.humidity = humidity;
        this.pressure = pressure;
        measurementsChanged();
    }
    public float getTemp() {
        return temp;
    }
    public void setTemp(float temp) {
        this.temp = temp;
    }
    public float getHumidity() {
        return humidity;
    }
    public void setHumidity(float humidity) {
        this.humidity = humidity;
    }
    public float getPressure() {
        return pressure;
    }
    public void setPressure(float pressure) {
        this.pressure = pressure;
    }
}

观察者接口定义:

/**
 * 观察者
 * @author Misout
 * @date 2018-03-25 12:40:28
 */
public interface Observer {
    /**
     * 更新温度、湿度、压强.
     * 缺点:将更新的参数值直接放入型参,如果未来要增减观察指标,将不可扩展。
     * @param temp 温度
     * @param humidity 湿度
     * @param pressure 压强
     */
    void update(float temp, float humidity, float pressure);
}

观察者具体实现类:

/**
 * @author Misout
 * @date 2018-03-25 13:33:58
 */
public class CurrentConditionsDisplay 
        implements Observer{
    private float temp;
    private float humidity;
    private float pressure;
    private Subject weatherData;
    /**
     * 注册到指定的Subject,进行监听
     * @param weatherData
     */
    public CurrentConditionsDisplay(Subject weatherData) {
        this.weatherData = weatherData;
        weatherData.registerObserver(this);
    }
    @Override
    public void update(float temp, float humidity, 
                float pressure) {
        this.temp = temp;
        this.humidity = humidity;
        this.pressure = pressure;
        display();
    }
    public void display() {
        System.out.println("Current conditions: " 
                + temp + "F degrees and " + humidity 
                + "% humidity and " + pressure 
                + " pressure");
    }
}

测试代码:

/**
 * @author Misout
 * @date 2018-03-25 13:37:49
 */
public class WeatherStationTest {
    public static void main(String[] args) {
        // 创建一个主题实现类:天气数据中心
        WeatherData weatherData = new WeatherData();
        // 创建一个观察者:当前天气显示器,并注册到天气中心
        CurrentConditionsDisplay currentDisplay = 
                new CurrentConditionsDisplay(weatherData);
        // 天气中心变更数据,自动通知到观察者
        weatherData.setMeasurements(37, 50, 39.4f);
    }
}

输出:

Current conditions: 37.0F degrees and 50.0% humidity and 39.4 pressure

6.JDK中的观察者模式

在JDK中已经提供了一套观察者模式的实现:Observable类相当于这里的Subject,Observer接口相当于这里的Observer接口

类图关系如下:

JDK观察者模式

值得注意的是,JDK中的主题类叫作Observable,是一个类,并非一个接口。这是一个缺陷,因为要想通过JDK的API实现观察者模式,必须继承Observable类,并不满足面对接口编程的设计原则,如果某个类想具有Observable和其他另一个父类的功能,就陷入两难的境地。因为Java类并不支持多继承。

7.总结

1、封装变化的部分。
2、多用组合、少用继承:JDK中的观察者模式给了一个教训。
3、针对接口编程,不针对实现编程。
4、为对象之间的依赖关系达到松耦合设计而努力。

推荐阅读:
设计模式(一)策略模式

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容