JS实现观察者模式详解
怎么入门文章编程?需要学习哪些知识点?这是新手们刚接触编程时常见的问题;下面golang学习网就来给大家整理分享一些知识点,希望能够给初学者一些帮助。本篇文章就来介绍《JS如何实现观察者模式》,涉及到,有需要的可以收藏一下
观察者模式的核心在于主题直接管理并通知观察者,而发布订阅模式通过事件中心解耦发布者与订阅者;在JavaScript中,该模式广泛应用于DOM事件、状态管理、实时数据更新等场景,其实现需注意内存泄漏、通知性能、错误处理及数据传递方式,确保系统解耦性与健壮性。

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的变化,然后根据变化重新渲染,这本质上也是一种观察行为。如果你自己实现一个简单的全局状态管理器,让组件注册监听某个状态的变化,那这就是观察者模式的典型应用。
还有一些场景,比如:
- 实时数据更新: 假设你有一个图表组件,需要实时展示后端推送的数据。后端数据一更新,主题(数据服务)就通知图表组件(观察者)去刷新。
- 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)是观察者模式的两种变体,各有优缺点。推送模式简单直接,但可能传递冗余数据;拉取模式更灵活,但观察者需要主动获取数据。具体用哪种,得看实际需求。
到这里,我们也就讲完了《JS实现观察者模式详解》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!
Twig中实现块的条件显示与隐藏方法
- 上一篇
- Twig中实现块的条件显示与隐藏方法
- 下一篇
- Greenshot截图模糊怎么调?清晰度设置教程
-
- 文章 · 前端 | 14小时前 | 前端 · 缓存 · Service Worker · 白屏 · 发布故障 · 缓存策略 前端白屏 Service Worker CacheStorage 资源404 发布回滚
- 前端发布后白屏复盘:Service Worker 缓存旧入口导致 JS 资源 404
- 469浏览 收藏
-
- 文章 · 前端 | 1天前 | 前端开发 · localStorage · 表格配置 · 用户偏好 · 后台系统 · 用户偏好 localStorage 前端表格 列配置 可见列 列宽保存
- 前端表格列设置刷新后丢失怎么办:可见列、列宽和顺序这样保存
- 351浏览 收藏
-
- 文章 · 前端 | 1天前 | 前端 · 接口排查 · 运维手册 · 性能告警 · 前端 AbortController 接口超时 Network瀑布图 降级回滚 线上告警
- 前端接口超时告警运行手册:从瀑布图到降级回滚
- 287浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ljg-skills
- ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
- 3072次使用
-
- MELO音乐
- MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
- 2832次使用
-
- UniScribe
- UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
- 2778次使用
-
- 剧云
- 剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
- 2996次使用
-
- 万象有声
- 万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
- 2952次使用
-
- JavaScript函数定义及示例详解
- 2025-05-11 502浏览
-
- CSS变量简化按钮悬停效果技巧
- 2026-05-31 501浏览
-
- JavaScript符号类型详解与应用
- 2026-05-31 501浏览
-
- HTML剪贴板复制粘贴怎么用
- 2026-05-26 501浏览
-
- data-*属性详解:HTML数据存储与DOM操作技巧
- 2026-05-25 501浏览

