设计模式-观察者模式

作者持续关注 WPS二次开发专题系列,持续为大家带来更多有价值的WPS开发技术细节,如果能够帮助到您,请帮忙来个一键三连,更多问题请联系我(QQ:250325397)

目录

定义

特点

使用场景

优缺点

[(1) 优点](#(1) 优点)

[(2) 缺点](#(2) 缺点)

模式结构

具体实现

实际应用


定义

观察者(Observer)模式的定义:指多个对象间存在一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。这种模式有时又称作发布-订阅模式模型-视图模式,它是对象行为型模式。

特点

定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。

使用场景

    • 对象间存在一对多关系,一个对象的状态发生改变会影响其他对象。
    • 当一个抽象模型有两个方面,其中一个方面依赖于另一方面时,可将这二者封装在独立的对象中以使它们可以各自独立地改变和复用。
    • 实现类似广播机制的功能,不需要知道具体收听者,只需分发广播,系统中感兴趣的对象会自动接收该广播。
    • 多层级嵌套使用,形成一种链式触发机制,使得事件具备跨域(跨越两种观察者类型)通知。

优缺点

(1) 优点
    • 降低了目标与观察者之间的耦合关系,两者之间是抽象耦合关系。符合依赖倒置原则。
    • 目标与观察者之间建立了一套触发机制。
(2) 缺点
    • 如果一个被观察者对象有很多的直接和间接的观察者的话,将所有的观察者都通知到会花费很多时间。
    • 如果在观察者和观察目标之间有循环依赖的话,观察目标会触发它们之间进行循环调用,可能导致系统崩溃。

模式结构

    • 抽象主题(Subject)角色:也叫抽象目标类,它提供了一个用于保存观察者对象的聚集类和增加、删除观察者对象的方法,以及通知所有观察者的抽象方法。
    • 具体主题(Concrete Subject)角色:也叫具体目标类,它实现抽象目标中的通知方法,当具体主题的内部状态发生改变时,通知所有注册过的观察者对象。
    • 抽象观察者(Observer)角色:它是一个抽象类或接口,它包含了一个更新自己的抽象方法,当接到具体主题的更改通知时被调用。
    • 具体观察者(Concrete Observer)角色:实现抽象观察者中定义的抽象方法,以便在得到目标的更改通知时更新自身的状态。

具体实现

主题

java 复制代码
/**
 * 主题
 */
public class Subject {

    private static Subject instance = new Subject();

    /**
     * 观察者的集合
     */
    private List<Observer> observers = new ArrayList<>();

    private Subject() {
    }

    public static Subject getInstance() {
        return instance;
    }

    /**
     * 订阅该主题
     */
    public synchronized void registerReceiver(Observer observer) {
        observers.add(observer);
    }

    /**
     * 取消订阅该主题
     */
    public synchronized void unregisterReceiver(Observer observer) {
        observers.remove(observer);
    }

    /**
     * 更新数据
     */
    public void onDataUpdate() {
        System.out.println("Subject update...");
        notifyObservers();
    }

    /**
     * 通知观察者
     */
    private void notifyObservers() {
        for (Observer observer : observers) {
            observer.update();
        }
    }

}

观察者

java 复制代码
/**
 * 观察者接口
 */
public interface Observer {
    /**
     * 更新
     */
    void update();
}

/**
 * 界面1
 */
public class FirstFragment implements Observer {

    /**
     * 界面创建
     */
    public void onCreate() {
        Subject.getInstance().registerReceiver(this);
    }

    /**
     * 界面销毁
     */
    public void onDestroy() {
        Subject.getInstance().unregisterReceiver(this);
    }

    @Override
    public void update() {
        System.out.println("FirstFragment update ui...");
    }
}

/**
 * 界面2
 */
public class SecondFragment implements Observer {

    /**
     * 界面创建
     */
    public void onCreate() {
        Subject.getInstance().registerReceiver(this);
    }

    /**
     * 界面销毁
     */
    public void onDestroy() {
        Subject.getInstance().unregisterReceiver(this);
    }

    @Override
    public void update() {
        System.out.println("SecondFragment update ui...");
    }
}

具体使用

java 复制代码
public class App {
    public static void main(String[] args) {
        //创建界面1
        FirstFragment firstFragment = new FirstFragment();
        firstFragment.onCreate();

        //创建界面2
        SecondFragment secondFragment = new SecondFragment();
        secondFragment.onCreate();

        //数据发生了变化
        Subject.getInstance().onDataUpdate();
    }
}

实际应用

  1. Android中的广播
  2. EventBus、Livedata、RxJava等
相关推荐
考虑考虑43 分钟前
JDK9中的dropWhile
java·后端·java ee
想躺平的咸鱼干1 小时前
Volatile解决指令重排和单例模式
java·开发语言·单例模式·线程·并发编程
hqxstudying1 小时前
java依赖注入方法
java·spring·log4j·ioc·依赖
·云扬·1 小时前
【Java源码阅读系列37】深度解读Java BufferedReader 源码
java·开发语言
Bug退退退1232 小时前
RabbitMQ 高级特性之重试机制
java·分布式·spring·rabbitmq
小皮侠2 小时前
nginx的使用
java·运维·服务器·前端·git·nginx·github
Zz_waiting.3 小时前
Javaweb - 10.4 ServletConfig 和 ServletContext
java·开发语言·前端·servlet·servletconfig·servletcontext·域对象
全栈凯哥3 小时前
02.SpringBoot常用Utils工具类详解
java·spring boot·后端
兮动人3 小时前
获取终端外网IP地址
java·网络·网络协议·tcp/ip·获取终端外网ip地址
呆呆的小鳄鱼3 小时前
cin,cin.get()等异同点[面试题系列]
java·算法·面试