前端开发模式—观察者模式

news/2025/3/16 15:39:32/文章来源:https://www.cnblogs.com/laozhenHome/p/18775064

观察者模式 (Observer Pattern)有时候也称为发布订阅模式,它的核心思想是定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。观察者模式是应用最广泛的开发模式之一,MVC 三层模式中的控制器就会观察视图并实时更新模型部分,Vue 中也有效地使用观察者模式来管理数据的响应式和视图的更新。

在很多文档中,观察者模式的主要角色被定义如下:

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

JS 中观察者模式的实现步骤如下:

1. 定义观察者类(订阅者)。

// 1. 定义观察者(订阅者)
class Observer {constructor(name) {this.name = name}// 观察者内部有一个更新方法,发布者通知观察者时,会调用观察者的更新方法update(value) {console.log(`${this.name}收到通知,通知内容为:${value}`)}
}

2. 定义被观察者类(主题、发布者、被观察对象)。

// 2. 定义被观察者(发布者、被观察对象)
// 发布者内部可以添加、删除、通知观察者
class Subject {constructor() {this.observers = [] // 维护观察者们的列表}// 添加观察者addObserver(observer) {this.observers.push(observer)}// 删除观察者removeObserver(observer) {this.observers = this.observers.filter(item => item !== observer)}// 通知观察者notifyObservers(value) {this.observers.forEach(observer => observer.update(value))}
}

3. 生成观察者实例对象和被观察者的实例对象。

// 创建观察者与发布者实例对象
const subject = new Subject()
const observer1 = new Observer('观察者1')
const observer2 = new Observer('观察者2')
const observer3 = new Observer('观察者3')

4. 将观察者实例对象们维护到被观察者实例对象中

// 将观察者维护到发布者中
subject.addObserver(observer1)
subject.addObserver(observer2)
subject.addObserver(observer3)

5. 被观察者状态改变时,通知观察者们;观察者们接收到通知后,进行相应的操作

subject.notifyObservers('明天房价上涨') // 观察者1收到通知,通知内容为:明天房价上涨  观察者2收到通知,通知内容为:明天房价上涨  观察者3收到通知,通知内容为:明天房价上涨subject.removeObserver(observer2)
subject.notifyObservers('明天房价下跌') // 观察者1收到通知,通知内容为:明天房价下跌  观察者3收到通知,通知内容为:明天房价下跌
点击查看完整代码
/*** 观察者模式*/// 观察者模式
// 1. 定义观察者(订阅者)
// 2. 定义被观察者(发布者、被观察对象)
// 3. 将观察者维护到发布者中
// 4. 被观察者状态改变时,通知观察者
// 5. 观察者接收到通知后,进行相应的操作// 1. 定义观察者(订阅者)
class Observer {constructor(name) {this.name = name}// 观察者内部有一个更新方法,发布者通知观察者时,会调用观察者的更新方法update(value) {console.log(`${this.name}收到通知,通知内容为:${value}`)}
}// 2. 定义被观察者(发布者、被观察对象)
// 发布者内部可以添加、删除、通知观察者
class Subject {constructor() {this.observers = [] // 维护观察者列表}// 添加观察者addObserver(observer) {this.observers.push(observer)}// 删除观察者removeObserver(observer) {this.observers = this.observers.filter(item => item !== observer)}// 通知观察者notifyObservers(value) {this.observers.forEach(observer => observer.update(value))}
}// 创建观察者与发布者实例对象
const subject = new Subject()
const observer1 = new Observer('观察者1')
const observer2 = new Observer('观察者2')
const observer3 = new Observer('观察者3')// 4. 将观察者维护到发布者中
subject.addObserver(observer1)
subject.addObserver(observer2)
subject.addObserver(observer3)// 5. 发布者下发通知,通知观察者
subject.notifyObservers('明天房价上涨') // 观察者1收到通知,通知内容为:明天房价上涨  观察者2收到通知,通知内容为:明天房价上涨  观察者3收到通知,通知内容为:明天房价上涨subject.removeObserver(observer2)
subject.notifyObservers('明天房价下跌') // 观察者1收到通知,通知内容为:明天房价下跌  观察者3收到通知,通知内容为:明天房价下跌

观察者模式的优点

解耦:观察者模式使得主题和观察者之间的耦合度降低,主题不需要知道观察者的具体实现。
动态注册:观察者可以在运行时动态注册和注销,灵活性高。
广播通知:主题可以一次性通知所有观察者,简化了事件处理逻辑。

观察者模式的缺点

性能问题:如果观察者数量较多,通知所有观察者可能会导致性能下降。
内存泄漏:如果观察者没有正确注销,可能会导致内存泄漏。
复杂性:在某些情况下,观察者模式可能会导致系统复杂性增加,特别是在观察者和主题之间的依赖关系较多时。

应用场景

自定义事件系统
组件间通信(如 Vue 的 EventBus)
实时数据更新(如 Websocket 消息推送)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/899822.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Tailscale subnet-on-android

前言啊哈,前段时间提了个issue,然后开发者那边回复了 再过了一段事件,合并了。Jan 17合并的,Feb 4发布的,那我倒要看看有没有新东西。不错不错,新功能加上了。 subnet有什么用 有人要问了,你安卓又不是路由器,搞个subnet干啥? 但是安卓可以开热点啊,尤其是在外面的网…

根据索引进行MySQL查询的简单优化

查询优化SQL让SQL尽量可以命中索引,可以提示查询的效率(但是数据库如果不走索引的速度较快,就不会去走索引)最左匹配法则由于联合索引中包括了多个列,那么对于这多个列的匹配就有一定的规则,就是最左匹配法则, 在使用联合索引时,必须满足从左边开始匹配索引列。假设现在…

FastAPI依赖注入:参数共享与逻辑复用

扫描二维码关注或者微信搜一搜:编程智域 前端至全栈交流与成长第一章:依赖注入核心原理 1.1 依赖树构建机制 from fastapi import Dependsdef auth_service():return OAuth2Scheme()def db_conn(auth: dict = Depends(auth_service)):return Database(creds=auth)@app.get(&q…

洛谷-P1449 后缀表达式

重操旧业~ 本菜鸡看到题就只会想到数组,但后来发现不可行,出去俩元素,还得进来一个元素,不好调o(╥﹏╥)o emm...但还是有所收获的,2转2,利用2-0,而不是直接强制int转换哦 同时也发现被题中举的例子迷惑了,数字可不一定是个位数! 1、此题采用栈和二叉树的后序遍历思想(…

DQL(查询语句)

简单查询 查询一个字段 select 字段名 from 表名; 查询多个字段 select 字段名1,字段名2,...,字段名 from 表名; 查询全部字段 ①在查询多个字段中,将所有的字段名写入 ②select * from 表名;————效率低,可读性差,因为将星号转化为所有字段名需要耗费一定时间,所…

PTA

​ 题目描述 小轿车中有一个系统随时监测四个车轮的胎压,如果四轮胎压不是很平衡,则可能对行车造成严重的影响。让我们把四个车轮 —— 左前轮、右前轮、右后轮、左后轮 —— 顺次编号为 1、2、3、4。本题就请你编写一个监测程序,随时监测四轮的胎压,并给出正确的报警信息。…

第二章

TempStr = input("请输入带有符号的温度值:") if isinstance(TempStr, str) and TempStr[-1] in [F, f]:C = int((float(TempStr[0:-1]) - 32) / 1.8)print(f"转换后的温度是{C}C") elif isinstance(TempStr, str) and TempStr[-1] in [C, c]:F = int(1.8…

window11家庭版 WSL上安装CentOS7

环境 Windows11家庭版 1. 安装WSL配置 1.1 Win命令模式开启虚拟化步骤点击查看代码 # 启用适用于 Linux 的 Windows 子系统:打开powershell并输入: dism.exe /online /enable-feature /featurename:Microsoft-Windows-Subsystem-Linux /all /norestart# 启用虚拟化:以管理员打…

探秘Transformer系列之(14)--- 残差网络和归一化

从零开始解析Transformer,目标是:(1) 解析Transformer如何运作,以及为何如此运作,让新同学可以入门;(2) 力争融入一些比较新的或者有特色的论文或者理念,让老鸟也可以有所收获。探秘Transformer系列之(14)--- 残差网络和归一化 目录探秘Transformer系列之(14)--- 残差…

省选联考2024游记

搬运自 洛谷博客 ,文章写于 2024.03.05“总有地上的生灵,敢于直面雷霆的威光。”全文共1169字。 Day -105 2023/11/18 noip考完人就崩了,第二题甚至都没写出正解,只有压线省一,大哭。 也基本注定这回是进不了省队了。 Day -5 2024/2/26 周一 开始停课 Day -4 2024/2/27 周…

day30 正则三剑客---awk

awk是什么 再谈三剑客grep,擅长单纯的查找或匹配文本内容 sed,更适合编辑、处理匹配到的文本内容 awk,更适合格式化文本内容,对文本进行复杂处理后、更友好的显示三个命令称之为Linux的三剑客 awk学完后的能力 以下部分内容需要结合shell编程对文本行数据提取数据字段 模式…

树上查分学习笔记+做题记录

树上查分 点的查分求路径 \(u-v\) 上的点被经过的次数. \(cnt[x]\) 表示点 \(x\) 被经过的次数. 核心代码:cnt[u]++; cnt[v]++; cnt[lca(u,v)]--; cnt[father[lca(u,v)]]--;A. 运输压力解法 树上查分板子题啊 #include <bits/stdc++.h> using namespace std; const int N…