《设计模式》学习笔记7——观察者模式

简介: 定义 观察者模式是使用频率最高的设计模式之一,也是最容易理解的设计模式之一,这种模式在生活中随处可见。观察者模式引用书中的定义如下: 观察者模式(Observer Pattern):定义对象之间的一种一对多依赖关系,使得每当一个对象状态发生改变时,其相关依赖对象皆得到通知并被自动更新。

定义

观察者模式是使用频率最高的设计模式之一,也是最容易理解的设计模式之一,这种模式在生活中随处可见。
观察者模式引用书中的定义如下:

观察者模式(Observer Pattern):定义对象之间的一种一对多依赖关系,使得每当一个对象状态发生改变时,其相关依赖对象皆得到通知并被自动更新。观察者模式的别名包括发布-订阅(PublishSubscribe) 模式、模型-视图(Model/View) 模式、源-监听器(Source/Listener)模式或从属者(Dependents) 模式。观察者模式是一种对象行为型模式

理解

观察者模式在生活中随处可见,路口的红绿灯和行人及车辆的关系是一种观察者模式,演员和观众是一种观察者模式,甚至于一个人说话,其他人听,也是一种观察者模式。
说白了,就是一个对象发出了某种信号,其他对象会根据这个对象立即作出某些动作或反应(什么也不做也是一种动作和反应)。
根据理解,我写了一个新闻网站根据中央气象台气象信息更新各自网站主页天气信息的小例子。

被观察者

在这里,中央气象台充当被观察者,各大网站充当观察者,首先我们定义一个被观察者的接口(但从实现观察者模式来说可以不用接口,但为了更好的拓展,这里使用接口编程)。

package patterntest.observerpattern.subject;
import java.util.ArrayList;
import java.util.List;
import patterntest.observerpattern.observer.WebSite;
public interface MyObservatory {
    /*
     * 存放观察者集合
     */
    List<WebSite> weblist = new ArrayList<WebSite>();
    /**
     * 增加新观察者
     * 
     * @param webSite
     */
    public void addWebSite(WebSite webSite);
    /**
     * 删除观察者
     * 
     * @param webSite
     */
    public void removeWebSite(WebSite webSite);
    /**
     * 发布天气信息
     */
    public void pubMsg();

}

在这个接口中有几个必要因素:

  1. 保存所有观察者对象的集合
  2. 新增观察者对象的方法
  3. 删除观察者对象的方法
  4. 发布信息的方法,也称为通知观察者的方法

然后写一个实现上述接口的实现类,把抽象方法实现:

package patterntest.observerpattern.subject;
import patterntest.observerpattern.observer.WebSite;
public class WeatherObservatory implements MyObservatory {
    @Override
    public void addWebSite(WebSite webSite) {
        weblist.add(webSite);
    }
    @Override
    public void removeWebSite(WebSite webSite) {
        weblist.remove(webSite);
    }
    @Override
    public void pubMsg() {
        for (WebSite webSite : weblist) {
            webSite.update();
        }
    }
}

add和remove方法应该不需要多说,就是向list集合加元素和删元素。重点在于pubMsg方法,也就是这里的发布天气信息的方法,这是观察者模式的核心所在。
在这个方法中,会遍历保存着所有观察者对象的集合,然后调用每个观察者对象的具体行为方法。当然了,这里边的逻辑可以自定义,未必就是通知所有观察者。

观察者

有了被观察者,接下来就是创建观察者,在上边的pubMsg方法中,我们调用了WebSite对象的update方法,但是现在还没有定义相应的对象。
观察者模式一般是一对多的关系,虽然说实际一对一也是没问题的,但是似乎就有些大材小用了。所以我们这里的观察者就使用多对象,这些对象都是观察者,所以有一个共同的父类:观察者接口。

package patterntest.observerpattern.observer;
/**
 * 观察者:网站接口
 * 
 * @author tzx
 * @date 2017年12月5日
 */
public interface WebSite {
    /**
     * 根据观察到的信息采取行动
     */
    public void update();
}

可以看到观察者接口的定义相对于被观察者就简单多了,只有一个update方法,也就是观察到被观察对象的状态变化后采取的行动。
观察者是一个抽象的说法,就需要有更具体的观察者,相对于这里的网站接口,我就给他两个观察者子类:新浪网和腾讯网。

package patterntest.observerpattern.observer;
public class Sina implements WebSite {
    @Override
    public void update() {
        System.out.println("新浪天气更新");
    }
}
package patterntest.observerpattern.observer;
public class Tencent implements WebSite {
    @Override
    public void update() {
        System.out.println("腾讯天气更新");
    }
}

上边两个观察者做的事太过于简单,所以就不多说了。接下来就模拟新浪网、腾讯网相继开始观察中央气象台的天气信息,然后中央气象台发布新的气象并通知两大网站。

import patterntest.observerpattern.observer.Sina;
import patterntest.observerpattern.observer.Tencent;
import patterntest.observerpattern.observer.WebSite;
import patterntest.observerpattern.subject.MyObservatory;
import patterntest.observerpattern.subject.WeatherObservatory;
import patterntest.observerpattern.subject.WindObservatory;
/**
 * 观察者模式测试
 * 
 * @author tzx
 * @date 2017年12月5日
 */
public class ObserverTest {
    public static void main(String[] args) {
        // 建立中央气象台
        MyObservatory observatory = new WeatherObservatory();
        // 创建新浪网
        WebSite sina = new Sina();
        // 新浪网开始观察中央气象台信息公告
        observatory.addWebSite(sina);
        // 创建腾讯网
        WebSite tencent = new Tencent();
        // 腾讯网开始观察中央气象台信息公告
        observatory.addWebSite(tencent);
        // 中央气象台发布新的天气信息
        int i = 0;
        do {
        observatory.pubMsg();
            i++;
            try {
                Thread.sleep(2000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        } while (i < 10);
    }
}

上边的测试方法中,先创建了一个中央气象台对象,然后创建新浪网和腾讯网对象并加入到气象观察队列。
之后循环10次调用气象台信息发布方法,新浪网对象和腾讯网对象就都会执行自己的update方法。
如果以后有新的观察者,例如搜狐网也要根据中央气象台的信息更新网站天气,那么只需要new一个搜狐网对象,然后调用addWebSite方法就行了。

要点

根据上边的实例和分析,观察者模式包含如下一些要点:

  1. 需要一个被观察者,也称作观察目标
  2. 观察目标中需要有一个保存所有观察者对象的集合
  3. 观察目标中需要有增加观察者对象的方法
  4. 观察目标中需要有删除观察者对象的方法
  5. 观察目标中需要有发布信息,也称作通知观察者对象的方法
  6. 观察者需要有一个提供给被观察者的方法

由上边总结的一些要点,应该很容易就写出一个观察者模式了,但是在观察者模式中还有一些需要特别说明的地方:

  1. java提供了对观察者模式的支持,java.util包中有关于观察者模式的两个接口Observer和Observable。所以要写一个观察目标的类,可以实现Observable接口;要写一个观察者的类,可以实现Observer接口。
  2. 观察者模式的实现逻辑上,可能与现实中的某些理解稍有差异。就如上边气象台的例子,原则上来说,气象台并不知道有哪些网站在观察气象信息,网站更新天气信息的主动权在网站手上。但是在代码实现观察者模式的时候,却是由观察目标来通知观察者的,观察目标那里需要保存所有观察者对象。

总结

观察者模式不论是理解还是具体实现都比较简单,可能也是因为简单,所以生活中随处可见,也可能是因为生活中随处可见,所以理解起来才显得简单。
观察者模式中保存了所有观察者对象,当使用抽象编程的时候,只需要维护一个观察者父类的集合,这样以后有了新的观察者,也不用改变观察目标,实现了一定程度的松耦合。
但是从上边示例中就可以看到,通知观察者的方法会遍历观察者对象集合,然后依次调用每个观察者对象的相应方法。所以如果这个观察者集合非常大,观察者非常多,或者相应的方法并不能很快结束。则必然影响整体的效率,也会出现数据延迟的问题。
同时,虽然可以通过参数传递的方式使观察者知道观察目标的状态,却无法知道观察目标的具体状态具体是如何变化的。

demo源码可在github下载:https://github.com/tuzongxun/mypattern

目录
相关文章
|
2月前
|
设计模式 监控 Java
Kotlin - 改良设计模式 - 观察者模式
Kotlin - 改良设计模式 - 观察者模式
58 3
|
27天前
|
设计模式 存储 供应链
前端必须掌握的设计模式——观察者模式
观察者模式(Observer Pattern)是一种行为型设计模式,实现了一种订阅机制。它包含两个角色:**观察者**(订阅消息、接收通知并执行操作)和**被观察者**(维护观察者列表、发送通知)。两者通过一对多的关系实现解耦,当被观察者状态改变时,会通知所有订阅的观察者。例如,商店老板作为被观察者,记录客户的需求并在商品到货时通知他们。前端应用中,如DOM事件注册、MutationObserver等也体现了这一模式。
|
2月前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
|
2月前
|
设计模式 消息中间件 搜索推荐
Java 设计模式——观察者模式:从优衣库不使用新疆棉事件看系统的动态响应
【11月更文挑战第17天】观察者模式是一种行为设计模式,定义了一对多的依赖关系,使多个观察者对象能直接监听并响应某一主题对象的状态变化。本文介绍了观察者模式的基本概念、商业系统中的应用实例,如优衣库事件中各相关方的动态响应,以及模式的优势和实际系统设计中的应用建议,包括事件驱动架构和消息队列的使用。
|
2月前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
48 1
|
2月前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
32 3
|
3月前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
39 9
|
3月前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
44 2
|
3月前
|
设计模式 监控 UED
设计模式之观察者模式
【10月更文挑战第12天】 观察者模式是一种行为型设计模式,定义了一对多的依赖关系,当一个对象状态改变时,所有依赖它的对象都会自动更新。主要由主题(被观察者)和观察者组成,实现对象间的松耦合,广泛应用于用户界面、事件驱动系统和数据监控等领域。
|
3月前
|
设计模式 Java Kotlin
Kotlin学习笔记 - 改良设计模式 - 迭代器模式
Kotlin学习笔记 - 改良设计模式 - 迭代器模式
40 2

热门文章

最新文章

  • 1
    设计模式转型:从传统同步到Python协程异步编程的实践与思考
    64
  • 2
    C++一分钟之-设计模式:工厂模式与抽象工厂
    54
  • 3
    《手把手教你》系列基础篇(九十四)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-下篇(详解教程)
    61
  • 4
    C++一分钟之-C++中的设计模式:单例模式
    79
  • 5
    《手把手教你》系列基础篇(九十三)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-上篇(详解教程)
    47
  • 6
    《手把手教你》系列基础篇(九十二)-java+ selenium自动化测试-框架设计基础-POM设计模式简介(详解教程)
    81
  • 7
    Java面试题:结合设计模式与并发工具包实现高效缓存;多线程与内存管理优化实践;并发框架与设计模式在复杂系统中的应用
    70
  • 8
    Java面试题:设计模式在并发编程中的创新应用,Java内存管理与多线程工具类的综合应用,Java并发工具包与并发框架的创新应用
    54
  • 9
    Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
    63
  • 10
    Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
    137