【对象行为模式】二十三天学会设计模式之观察者模式

简介: 【对象行为模式】二十三天学会设计模式之观察者模式

一、什么是观察者模式

观察者一般可以看做是第三者,比如在学校上自习的时候,大家肯定都有过交头接耳、各种玩耍的经历,这时总会有一个“放风”的小伙伴,当老师即将出现时及时“通知”大家老师来了。再比如,拍卖会的时候,大家相互叫价,拍卖师会观察最高标价,然后通知给其它竞价者竞价,这就是一个观察者模式。

 

对于观察者模式而言,有两个对象,一个是观察者,一个是被观察者。

 

观察者模式(Observer),又叫发布-订阅模式(Publish/Subscribe),

定义对象间一种一对多的依赖关系,使得每当一个对象改变状态,则所有依赖于它的对象都会得到通知并自动更新。UML结构图如下:

其中,各属性的定义:

  • Subject:被观察者
  • Observer:抽象观察者接口
  • ConcreteObserver:具体观察者,实现接口

二、为什么要用观察者模式?

大家在生活中可能都遇到这种问题:当孩子哭的时候,爸爸、妈妈应该第一时间收到消息,并且做出响应的对策。

如果我们不采取观察者模式,我们写出的代码可能是这种的:

这种代码在DadMum等观察者少量的时候还比较好用,当观察者的数量多起来,我们的类就会变得十分臃肿。

还有一个比较大的问题,对于观察者来说,不一定监控一个被观察者,可能同时监控多个被观察者,这样的话,我们上述简单的架构就会变得累赘。


原始的观察者模式在传递消息的时候,是通过观察者类和被观察者类之间的耦合通知,我们将消息的通知改为事件发送,也就是观察者和被观察者通过事件来进行通信。

三、如何实现观察者模式?

我们还是用上面小孩哭的例子,来进行一个架构图的绘制:

整体的思路:当我们的小孩哭了之后,会发送一个哭泣的消息事件,而我们的 observer 会监听这个事件并做出响应

Child:被观察者对象,发送哭泣的事件

class Child {
    private Boolean cry = false;
    private List<Observer> observers = new ArrayList<>();
    {
        observers.add(new Dad());
        observers.add(new Mum());
        observers.add(new Dog());
    }
    public boolean isCry() {
        return cry;
    }
    public void wakeUp() {
        cry = true;
        // 创建消息事件
        WakeUpEvent event = new WakeUpEvent(System.currentTimeMillis(), "bed", this);
        // 发送消息事件
        for (Observer observer : observers) {
            observer.actionOnWakeUp(event);
        }
    }
}

WakeUpEvent:事件发送事件、地点、事件源对象

class WakeUpEvent {
    long timestamp;
    String loc;
    // 事件源对象本身
    Child source;
    public WakeUpEvent(long timestamp, String loc, Child source) {
        this.timestamp = timestamp;
        this.loc = loc;
        this.source = source;
    }
}

Observer:观察者的接口

interface Observer {
    void actionOnWakeUp(WakeUpEvent event);
}

Dad:实现观察者接口,监听事件

class Dad implements Observer {
    public void feed() {
        System.out.println("dad feed");
    }
    @Override
    public void actionOnWakeUp(WakeUpEvent event) {
        feed();
    }
}

Test:测试

  public static void main(String[] args) {
        Child child = new Child();
        child.wakeUp();
        // dad feed
    // Mum bao
    // Dog wangwangwang
    }

四、总结

当我们想要一个对象状态改变,其他对象状态也跟着改变时,如果我们使用原始类耦合的做法,会造成我们的类显着特别的臃肿和难以维护。

这个时候,我们利用 观察者模式+事件机制,可以让我们整个系统的耦合度大大降低,并且在后续维护的过程中,也便于维护。

但是观察者也有一定的弊端,比如:

  • 我们的观察者并不知道被观察者做出了怎么样的改变,仅仅知道被观察者进行了变化
  • 一定避免循环观察,防止观察者和被观察者之间相互引用

相关文章
|
3月前
|
设计模式 监控 安全
设计模式 | 观察者模式
设计模式 | 观察者模式
18 0
|
3月前
|
设计模式 前端开发 数据中心
设计模式之观察者模式
设计模式之观察者模式
|
2月前
|
设计模式 前端开发 JavaScript
观察者模式 vs 发布-订阅模式:两种设计模式的对决!
欢迎来到前端入门之旅!这个专栏是为那些对Web开发感兴趣、刚刚开始学习前端的读者们打造的。无论你是初学者还是有一些基础的开发者,我们都会在这里为你提供一个系统而又亲切的学习平台。我们以问答形式更新,为大家呈现精选的前端知识点和最佳实践。通过深入浅出的解释概念,并提供实际案例和练习,让你逐步建立起一个扎实的基础。无论是HTML、CSS、JavaScript还是最新的前端框架和工具,我们都将为你提供丰富的内容和实用技巧,帮助你更好地理解并运用前端开发中的各种技术。
|
20天前
|
设计模式 监控 Java
设计模式 - 观察者模式(Observer):Java中的战术与策略
【4月更文挑战第7天】观察者模式是构建可维护、可扩展系统的关键,它在Java中通过`Observable`和`Observer`实现对象间一对多的依赖关系,常用于事件处理、数据绑定和同步。该模式支持事件驱动架构、数据同步和实时系统,但需注意避免循环依赖、控制通知粒度,并关注性能和内存泄漏问题。通过明确角色、使用抽象和管理观察者注册,可最大化其效果。
|
1月前
|
设计模式 存储 Java
【设计模式】观察者模式
【设计模式】观察者模式
|
3天前
|
设计模式 JavaScript Java
[设计模式Java实现附plantuml源码~行为型] 对象状态及其转换——状态模式
[设计模式Java实现附plantuml源码~行为型] 对象状态及其转换——状态模式
|
30天前
|
设计模式 算法 中间件
【C++ 可调用对象的应用】C++设计模式与现代编程技巧:深入可调用对象的世界
【C++ 可调用对象的应用】C++设计模式与现代编程技巧:深入可调用对象的世界
114 1
|
3月前
|
设计模式 uml
设计模式-适配器模式(对象适配器)
设计模式-适配器模式(对象适配器)
19 0
|
3月前
|
设计模式 Java Spring
设计模式之观察者模式
设计模式之观察者模式
26 0
|
3月前
|
设计模式 算法 自动驾驶
常见的设计模式(模板与方法,观察者模式,策略模式)
随着时间的推移,软件代码越来越庞大,随着而来的就是如何维护日趋庞大的软件系统。在面向对象开发出现之前,使用的是面向过程开发来设计大型的软件程序,面向过程开发将软件分成一个个单独的模块,模块之间使用函数进行组合,最后完成系统的开发,每次需要修改软件,如果不涉及好各个模块的关系,就会导致软件系统难以维护,从而导致软件变得不可使用。面向对象方法用对象模拟问题域中的实体,以对象间的联系刻画实体间联系
65 2