设计模式之观察者模式

简介: 一、背景   随着工作时间的越来越长,发现对设计模式缺失的坏处越来越明显,但是当你知道某种设计模式的实现方式以后,你会发现,其实工作中早已经玩过这些东西,但是你之前并不知道它属于设计模式的一种,今天就先介绍一种设计模式:观察者模式,然后我们也手动实现观察者模式以加深印象。

一、背景

  随着工作时间的越来越长,发现对设计模式缺失的坏处越来越明显,但是当你知道某种设计模式的实现方式以后,你会发现,其实工作中早已经玩过这些东西,但是你之前并不知道它属于设计模式的一种,今天就先介绍一种设计模式:观察者模式,然后我们也手动实现观察者模式以加深印象。

二、模式介绍

  观察者模式是对象的行为模式,又叫发布-订阅(Publish/Subscribe)模式、模型-视图(Model/View)模式、源-监听器(Source/Listener)模式或者从属者(Dependents)模式。观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己。

  应用场景:一个软件系统中常常要求在某一个对象的状态发生变化的时候,某些和它关联的其他的对象做出相应的改变。做到这一点的设计方案有很多,但是为了使系统能够易于复用,应该选择低耦合的设计方案。减少对象之间的耦合有利于系统的复用,但是同时设计时需要使这些低耦合的对象之间能够维持行为的协调一致,保证高度的协作。观察者模式是满足这一要求的各种设计方案中最重要的一种。

三、观察者模式结构和组成

  

  由上图可知,观察者模式所涉及的角色有:

  1.抽象主题角色(Subject):抽象主题角色把所有对观察者对象的引用保存在一个集合(比如Vector)里,然后每个主题都可以有任何数量的观察者。抽象主题提供的接口有:增加观察者对象、删除观察者对象、通知所有的观察者对象、获取观察者数量、设置改变。

  2.具体主题角色(ConcreteSubject): 将有关状态存入具体主题对象,在具体主题的内部状态改变时,给所有注册过的观察者发送通知。

  3.抽象观察者角色(Observer):为所有的具体观察者定义一个接口,订阅的主题改变时更新自己,这个接口叫做更新接口。

  4.具体观察者角色(ConcreteObserver):存储与主题自恰的状态。具体观察者角色实现抽象观察者角色所要求的更新接口,以便使本身的状态与主题的状态向协调。如有需要,具体观察者角色可以保持一个指向具体主题对象的引用。

四、代码实现

1.抽象观察者角色Observer.java接口

package com.hafiz.www.observerPattern;

/**
 * Desc:观察者接口
 * Created by hafiz.zhang on 2017/7/27.
 */
public interface Observer {

    /**
     * 已订阅的主题发生改变时调用此方法来更新自己
     *
     * @param obj 当前正在发生改变的可观察者对象
     *
     * @param args 参数
     */
    void update(Observable obj, Object args);
}

2.抽象主题角色Observable.java类

package com.hafiz.www.observerPattern;

import java.util.Vector;

/**
 * Desc: 抽象可被观察者类
 * Created by hafiz.zhang on 2017/7/27.
 */
public abstract class Observable {

    private Vector<Observer> observers;     // 已订阅的观察者集合
    private boolean changed = false;        // 状态

    public Observable() {
        observers = new Vector<>();
    }

    public synchronized void addObserver(Observer observer) {
        if (observer == null) {
            throw new NullPointerException();
        }
        if (!observers.contains(observer)) {
            observers.addElement(observer);
        }
    }

    public synchronized void removeObserver(Observer observer) {
        observers.remove(observer);
    }

    public synchronized void setChanged() {
        changed = true;
    }

    public synchronized void clearChanged() {
        changed = false;
    }

    public void notifyObservers(Object args) {
        Observer[] observersLocal;
        synchronized (this) {
            if (!changed) {
                return;
            }
            observersLocal = new Observer[observers.size()];
            observersLocal = observers.toArray(observersLocal);
            clearChanged();
        }
        for (int i = 0; i < observersLocal.length; i++) {
            observersLocal[i].update(this, args);
        }
    }

    public void notifyObservers() {
        notifyObservers(null);
    }

    public synchronized void deleteAllObserver() {
        observers.removeAllElements();
    }

    public synchronized int countObservers() {
        return observers.size();
    }
}

3.具体观察者实现类ConcreteObserver.java类

package com.hafiz.www.observerPattern;

/**
 * Desc:具体观察者实现类
 * Created by hafiz.zhang on 2017/7/27.
 */
public class ConcreteObserver implements Observer {
    private String status;

    @Override
    public void update(Observable obj, Object args) {
        System.out.println("Observable:" + obj.getClass().getName() + " status has changed");
        this.status = (String)args;
        System.out.println("After changed ConcreteObserver`s status is : " + status);
    }
}

4.具体主题实现类ConcreteSubject.java类

package com.hafiz.www.observerPattern;

/**
 * Desc:具体主题实现类
 * Created by hafiz.zhang on 2017/7/27.
 */
public class ConcreteSubject extends Observable {
    private String status = "originalStatus";

    public ConcreteSubject(Observer observer) {
        // 注册观察者
        addObserver(observer);
    }

    public void changeStatus(String status) {
        System.out.println("Before changed ConcreteSubject`s status is : " + this.status);
        this.status = status;
        System.out.println("After changed ConcreteSubject`s status is : " + this.status);
        setChanged();
        notifyObservers(this.status);
    }
}

5.单元测试类

package com.hafiz.www;

import com.hafiz.www.observerPattern.ConcreteObserver;
import com.hafiz.www.observerPattern.ConcreteSubject;
import org.junit.Test;

/**
 * Desc:设计模式demo单元测试类
 * Created by hafiz.zhang on 2017/7/27.
 */
public class DesignPatternTest {

    @Test
    public void testObserverPattern() {
        ConcreteObserver observer = new ConcreteObserver();
        ConcreteSubject subject = new ConcreteSubject(observer);
        subject.changeStatus("new Status");
    }
}

6.运行结果截图:

7.说明和扩展

  在观察者模式中,其实又细分为推模型和拉模型,所谓推模型是指,在主题类发生改变的时候,调用观察者update()方法直接把数据通过args参数传过去,不管观察者是否真的需要;而拉模型是调用update()方法时,把自己传递给观察者,观察者需要什么数据,它自己从主题对象中获取。在我看来拉模型更加的易于扩展。

  其实,在java.util包里面,jdk默认也给我们提供了这样实现,其中Observer.java是观察者抽象类,Observable.java是主题抽象类,我们可以拿来使用,其中的实现和本文的差别不大。使用时,我们只需要基于这两个类进行实现就可以了很简单。推荐使用!

  文中代码github地址:https://github.com/hafizzhang/DesignPattern.git

五、总结

  通过本文,我们就了解了什么叫观察者模式,以及怎么实现该模式,以及该模式的特点。以后的工作中,我们很好的结合具体场景来使用,可以说收获满满!喜欢请点赞。

相关文章
|
4月前
|
设计模式 PHP
php设计模式--观察者模式(4.1)面向过程完成页面内容切换
php设计模式--观察者模式(4.1)面向过程完成页面内容切换
29 0
|
24天前
|
设计模式 存储 前端开发
【十四】设计模式~~~行为型模式~~~观察者模式(Java)
文章详细介绍了观察者模式(Observer Pattern),这是一种对象行为型模式,用于建立对象之间的一对多依赖关系。当一个对象状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。文中通过交通信号灯与汽车的案例以及多人联机对战游戏的设计方案,阐述了观察者模式的动机和应用场景。接着,文章介绍了观察者模式的结构、角色、优点、缺点以及适用情况,并通过代码示例展示了如何在Java中实现观察者模式。此外,还探讨了观察者模式在MVC架构中的应用以及Java中对观察者模式的支持。
【十四】设计模式~~~行为型模式~~~观察者模式(Java)
|
4月前
|
设计模式 监控 Java
设计模式 - 观察者模式(Observer):Java中的战术与策略
【4月更文挑战第7天】观察者模式是构建可维护、可扩展系统的关键,它在Java中通过`Observable`和`Observer`实现对象间一对多的依赖关系,常用于事件处理、数据绑定和同步。该模式支持事件驱动架构、数据同步和实时系统,但需注意避免循环依赖、控制通知粒度,并关注性能和内存泄漏问题。通过明确角色、使用抽象和管理观察者注册,可最大化其效果。
96 2
|
1月前
|
设计模式 安全 Go
[设计模式]行为型模式-观察者模式
[设计模式]行为型模式-观察者模式
|
1月前
|
设计模式 Go
go 设计模式之观察者模式
go 设计模式之观察者模式
|
2月前
|
设计模式 安全 Java
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
60 1
|
2月前
|
设计模式 缓存
iLogtail设计模式问题之观察者模式在iLogtail中是如何应用的
iLogtail设计模式问题之观察者模式在iLogtail中是如何应用的
|
3月前
|
设计模式 消息中间件 存储
跟着GPT学设计模式之观察者模式
观察者模式是一种行为型设计模式,它定义了对象之间的一对多依赖关系,使得当一个对象的状态发生改变时,其依赖对象都能够收到通知并自动更新。一般情况下,被依赖的对象叫作被观察者(Observable),依赖的对象叫作观察者(Observer)。
26 1
|
3月前
|
设计模式 存储 Java
Java设计模式之观察者模式详解
Java设计模式之观察者模式详解
|
3月前
|
设计模式
观察者模式-大话设计模式
观察者模式-大话设计模式