0

0

JS如何实现观察者模式

畫卷琴夢

畫卷琴夢

发布时间:2025-08-13 11:12:02

|

977人浏览过

|

来源于php中文网

原创

观察者模式的核心在于主题直接管理并通知观察者,而发布订阅模式通过事件中心解耦发布者与订阅者;在javascript中,该模式广泛应用于dom事件、状态管理、实时数据更新等场景,其实现需注意内存泄漏、通知性能、错误处理及数据传递方式,确保系统解耦性与健壮性。

JS如何实现观察者模式

JavaScript中实现观察者模式,核心在于构建一个“主题”(Subject)对象,它负责维护一个订阅者(Observer)列表,并在自身状态发生变化时,遍历这个列表并通知所有已注册的订阅者。说白了,就是让感兴趣的对象(观察者)去“监听”另一个对象(主题)的变化,一旦有变动,主题就会告诉它们。

class Subject {
    constructor() {
        this.observers = []; // 存储所有观察者
    }

    /**
     * 添加一个观察者
     * @param {Function} observer - 观察者函数或对象,通常是一个回调函数
     */
    addObserver(observer) {
        if (typeof observer === 'function' && !this.observers.includes(observer)) {
            this.observers.push(observer);
        } else {
            // 我个人觉得,这里加个简单的错误提示或者日志会更好,
            // 毕竟有时候传入的可能不是函数,或者重复添加了
            console.warn('Observer must be a unique function.');
        }
    }

    /**
     * 移除一个观察者
     * @param {Function} observer - 要移除的观察者
     */
    removeObserver(observer) {
        this.observers = this.observers.filter(obs => obs !== observer);
    }

    /**
     * 通知所有观察者状态已更新
     * @param {*} data - 传递给观察者的数据
     */
    notify(data) {
        // 有时候我会考虑是否要异步通知,避免一个耗时观察者阻塞其他,
        // 但对于多数前端场景,同步通知也挺常见。
        this.observers.forEach(observer => {
            try {
                observer(data);
            } catch (error) {
                // 这里捕获一下观察者执行时的错误,避免一个观察者出错导致整个通知链中断
                console.error('Error notifying observer:', error);
            }
        });
    }
}

// 示例用法:
const mySubject = new Subject();

// 定义一些观察者
const observerA = (data) => {
    console.log('观察者A收到通知:', data);
};

const observerB = (data) => {
    console.log('观察者B收到通知,数据是:', data);
};

const observerC = (data) => {
    console.log('观察者C处理数据:', JSON.stringify(data));
};

// 注册观察者
mySubject.addObserver(observerA);
mySubject.addObserver(observerB);
mySubject.addObserver(observerC);

console.log('--- 第一次状态变化 ---');
mySubject.notify({ message: 'Hello from Subject!', timestamp: Date.now() });

// 移除一个观察者
mySubject.removeObserver(observerB);

console.log('--- 第二次状态变化(B已移除)---');
mySubject.notify('新的消息');

// 尝试添加一个非函数或重复的
mySubject.addObserver({}); // 会有警告
mySubject.addObserver(observerA); // 不会重复添加

观察者模式与发布订阅模式有何不同?

这真的是一个老生常谈的问题,但它确实很重要,因为两者概念上很接近,但在实现和职责划分上有所区别。在我看来,最核心的区别在于它们之间是否存在一个“中间人”。观察者模式中,主题(Subject)是直接知道并管理着所有观察者(Observer)的。它维护一个观察者列表,并在状态改变时直接调用这些观察者的方法。这是一种“一对多”的直接依赖关系。就像一个明星(主题)直接管理着他的粉丝俱乐部(观察者),有新动态就直接发给所有粉丝。

而发布订阅模式(Pub/Sub)则引入了一个“事件中心”或“消息代理”(Broker/Event Bus)。发布者(Publisher)不直接知道订阅者(Subscriber),它只负责向事件中心发布消息;订阅者也不直接知道发布者,它只向事件中心订阅感兴趣的消息。所有的通信都通过这个中间人来完成。这就像出版社(发布者)把书交给书店(事件中心),读者(订阅者)去书店买书,出版社和读者之间并不直接打交道。

从解耦程度上看,发布订阅模式的解耦更彻底,因为它消除了发布者和订阅者之间的直接依赖。主题(发布者)和观察者(订阅者)之间完全互不干涉。这在大型应用中特别有用,可以避免组件之间形成复杂的网状依赖。观察者模式虽然也实现了观察者和主题的解耦(观察者不需要知道主题的具体实现,只需要知道它有

notify
方法),但主题本身还是需要管理观察者的列表,这种耦合是存在的。所以,选择哪种模式,往往取决于你对系统解耦程度的需求以及复杂性考量。

在哪些实际场景中,观察者模式能发挥作用?

观察者模式在前端开发中简直是无处不在,只是我们可能不总用“观察者模式”这个词来称呼它。最直观的例子就是DOM事件处理。当你在一个按钮上添加

addEventListener
监听点击事件时,这个按钮就是主题,你的回调函数就是观察者。按钮被点击时,它会通知所有注册的监听器。这不就是典型的观察者模式吗?

再比如,单页应用中的状态管理。虽然像Redux这样的库更多地被认为是发布订阅模式(因为它有一个中央的store作为事件中心),但其内部某些机制,比如组件订阅store的变化,然后根据变化重新渲染,这本质上也是一种观察行为。如果你自己实现一个简单的全局状态管理器,让组件注册监听某个状态的变化,那这就是观察者模式的典型应用。

DreamStudio
DreamStudio

SD兄弟产品!AI 图像生成器

下载

还有一些场景,比如:

  • 实时数据更新: 假设你有一个图表组件,需要实时展示后端推送的数据。后端数据一更新,主题(数据服务)就通知图表组件(观察者)去刷新。
  • UI组件间的通信: 当一个表单输入框的值改变时,可能需要另一个显示区域同步更新。输入框作为主题,显示区域作为观察者。
  • 游戏开发: 玩家角色的生命值、分数等属性变化时,可以通知UI显示、音效播放等观察者。

我觉得,只要是涉及到“当A发生变化时,B、C、D需要做出响应”这种需求,观察者模式就值得考虑。它提供了一种清晰、可维护的方式来处理这种一对多的依赖关系。

实现观察者模式时,有哪些常见的陷阱或优化考量?

实现观察者模式,虽然概念上简单,但在实际应用中还是有些细节需要注意,不然可能会引入一些问题。

一个比较常见的陷阱是内存泄漏。如果观察者没有被正确地从主题中移除,即使观察者本身已经不再被使用了(比如对应的DOM元素被移除了),主题仍然会持有对它的引用。这意味着垃圾回收器无法回收这个观察者,导致内存占用持续增加。这在单页应用中尤其常见,页面切换或者组件销毁时,忘记取消订阅是很要命的。所以,我的经验是,在组件生命周期结束时(比如React的

componentWillUnmount
useEffect
的返回函数中),一定要记得调用
removeObserver

另一个是通知的顺序和性能。如果有很多观察者,或者某个观察者的回调函数执行时间很长,同步通知可能会阻塞主线程,导致UI卡顿。这时可以考虑异步通知,比如使用

setTimeout
requestAnimationFrame
或者微任务(
Promise.resolve().then(...)
)来调度观察者的执行。这样可以把耗时操作放到事件循环的下一轮,避免阻塞。但异步通知也会带来额外的复杂性,比如如何确保通知的顺序,以及如何处理通知失败的情况。

再来就是错误处理。在我的示例代码里,我加了一个

try-catch
块来包裹观察者的执行。这是很重要的,因为如果一个观察者在执行时抛出了未捕获的错误,它可能会中断整个通知链,导致后续的观察者无法收到通知。捕获错误并记录下来,可以确保系统的健壮性。

最后,还有参数传递的考量。

notify
方法通常会传递一些数据给观察者。这些数据应该包含观察者所需的所有信息,但也要避免传递过大的对象,否则会增加内存开销和数据序列化的负担。有时候,一个简单的ID或者一个表示变化类型的枚举值就足够了,观察者可以根据这个ID再去主题那里拉取更详细的数据。这种“拉取”模式(Pull Model)与“推送”模式(Push Model)是观察者模式的两种变体,各有优缺点。推送模式简单直接,但可能传递冗余数据;拉取模式更灵活,但观察者需要主动获取数据。具体用哪种,得看实际需求。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

766

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

766

2023.08.10

js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

531

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

576

2023.07.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

js是什么意思
js是什么意思

JS是JavaScript的缩写,它是一种广泛应用于网页开发的脚本语言。JavaScript是一种解释性的、基于对象和事件驱动的编程语言,通常用于为网页增加交互性和动态性。它可以在网页上实现复杂的功能和效果,如表单验证、页面元素操作、动画效果、数据交互等。

6283

2023.08.17

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

493

2023.09.01

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
JavaScript 基础加强视频教程
JavaScript 基础加强视频教程

共73课时 | 17.5万人学习

Git工具使用小知识
Git工具使用小知识

共38课时 | 17.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号