当前位置: 首页 > article >正文

Golang学习笔记_41——观察者模式

Golang学习笔记_38——享元模式
Golang学习笔记_39——策略模式
Golang学习笔记_40——模版方法模式


文章目录

    • 一、核心概念
      • 1. 定义
      • 2. 解决的问题
      • 3. 核心角色
      • 4. 类图
    • 二、特点分析
    • 三、适用场景
      • 1. 股票价格监控系统
      • 2. 物联网设备状态监控
      • 3. 电商订单状态通知
    • 四、Go语言实现示例
      • 完整实现代码
      • 执行结果
    • 五、高级应用
      • 1. 异步通知机制(参考网页10)
      • 2. 事件过滤机制
    • 六、与其他模式对比
    • 七、实现建议
    • 八、典型应用


一、核心概念

1. 定义

观察者模式是一种行为型设计模式,建立对象间的一对多依赖关系,当一个对象(主题)状态改变时,自动通知所有依赖对象(观察者)。其核心特点包括:

  • 松耦合设计:主题与观察者通过接口交互()
  • 动态订阅:运行时增减观察者()
  • 广播通知:状态变化自动触发多对象更新()

2. 解决的问题

  • 状态同步:多个对象需实时获取核心对象状态变化
  • 事件驱动:构建解耦的事件响应系统()
  • 资源监控:如CPU使用率报警()

3. 核心角色

角色作用
Subject维护观察者列表,定义注册/移除/通知方法()
Observer定义更新接口,接收主题通知()
ConcreteSubject实现具体状态管理和通知逻辑
ConcreteObserver实现业务响应逻辑

4. 类图

观察者模式类图

@startuml
interface Subject {+ Register(Observer)+ Remove(Observer)+ Notify()
}interface Observer {+ Update()
}class ConcreteSubject {- state: int+ SetState(int)
}class ConcreteObserver {+ Update()
}Subject <|-- ConcreteSubject
Observer <|-- ConcreteObserver
ConcreteSubject "1" *-- "0..*" Observer
@enduml

二、特点分析

优点

  1. 解耦设计:主题与观察者无直接依赖()
  2. 动态扩展:支持运行时增减观察者()
  3. 事件驱动:适用于异步处理场景()

缺点

  1. 通知顺序:多个观察者的执行顺序不可控()
  2. 循环触发:不当设计可能引发循环调用()
  3. 性能开销:大量观察者时通知效率降低()

三、适用场景

1. 股票价格监控系统

// 股票主题
type StockSubject struct {observers []Observerprice     float64
}func (s *StockSubject) Register(o Observer) {s.observers = append(s.observers, o)
}func (s *StockSubject) Notify() {for _, o := range s.observers {o.Update(s.price)}
}// 手机APP观察者
type MobileApp struct{}func (m *MobileApp) Update(price float64) {fmt.Printf("手机端收到股价更新: %.2f\n", price)
}

2. 物联网设备状态监控

// 设备状态主题
type DeviceMonitor struct {observers []Observerstatus    string
}func (d *DeviceMonitor) SetStatus(status string) {d.status = statusd.Notify()
}// 运维系统观察者
type OpsSystem struct{}func (o *OpsSystem) Update(status string) {if status == "ERROR" {fmt.Println("触发告警:设备异常")}
}

3. 电商订单状态通知

type OrderSubject struct {observers []Observerstate     string
}func (o *OrderSubject) Ship() {o.state = "SHIPPED"o.Notify()
}// 短信通知观察者
type SMSNotifier struct{}func (s *SMSNotifier) Update(state string) {if state == "SHIPPED" {fmt.Println("发送物流短信通知")}
}

四、Go语言实现示例

示例类图

完整实现代码

package observer_demoimport ("fmt""sync""time"
)// OrderEvent 事件结构体
type OrderEvent struct {OrderId     stringAmount      float64PaymentTime time.TimeUserID      string
}// Observer 观察者接口
type Observer interface {Handle(event OrderEvent) errorName() string
}// OrderSubject 主题接口
type OrderSubject interface {Subscribe(observer Observer)UnSubscribe(observer Observer)Notify(event OrderEvent)
}// OrderService 具体主题实现
type OrderService struct {Observer sync.Map // 使用现成安全的Map存储观察者wg       sync.WaitGroup
}func (s *OrderService) Subscribe(observer Observer) {s.Observer.Store(observer.Name(), observer)
}func (s *OrderService) UnSubscribe(observer Observer) {s.Observer.Delete(observer.Name())
}func (s *OrderService) Notify(event OrderEvent) {s.Observer.Range(func(key, value interface{}) bool {observer := value.(Observer)s.wg.Add(1)go func(obs Observer) {defer s.wg.Done()// 带超时控制的处理done := make(chan struct{})go func() {defer close(done)if err := retry(3, time.Second, func() error {return obs.Handle(event)}); err != nil {fmt.Printf("[Error] %s处理失败: %v\n", obs.Name(), err)}}()select {case <-done:fmt.Printf("[Info] %s处理完成\n", obs.Name())case <-time.After(5 * time.Second):fmt.Printf("[Error] %s处理超时\n", obs.Name())}}(observer)return true})s.wg.Wait()
}// 重试函数
func retry(attempts int, sleep time.Duration, fn func() error) error {if err := fn(); err != nil {if attempts--; attempts > 0 {time.Sleep(sleep)return retry(attempts, sleep, fn)}return err}return nil
}// InventoryObserver 具体观察者实现
type InventoryObserver struct {
}func (i *InventoryObserver) Name() string {return "InventoryObserver"
}func (i *InventoryObserver) Handle(event OrderEvent) error {// 模拟库存扣减操作fmt.Printf("库存系统:订单%s扣减库存,用户%s,金额%.2f\n",event.OrderId, event.UserID, event.Amount)return nil
}// LogisticsObserver 具体观察者实现:物流业务
type LogisticsObserver struct {
}func (l *LogisticsObserver) Name() string {return "LogisticsObserver"
}func (l *LogisticsObserver) Handle(event OrderEvent) error {// 模拟物流操作fmt.Printf("物流系统:订单%s发货,用户%s,金额%.2f\n",event.OrderId, event.UserID, event.Amount)return nil
}// UserObserver 具体观察者实现,用户服务
type UserObserver struct {
}func (u *UserObserver) Name() string {return "UserObserver"
}func (u *UserObserver) Handle(event OrderEvent) error {// 模拟用户服务操作fmt.Printf("用户服务:订单%s创建成功,用户%s,金额%.2f\n",event.OrderId, event.UserID, event.Amount)return nil
}func test() {// 创建主题orderService := &OrderService{}// 创建观察者inventoryObserver := &InventoryObserver{}logisticsObserver := &LogisticsObserver{}userObserver := &UserObserver{}// 注册观察者orderService.Subscribe(inventoryObserver)orderService.Subscribe(logisticsObserver)orderService.Subscribe(userObserver)// 模拟支付成功事件event := OrderEvent{OrderId:     "20230303123456",Amount:      2999.00,PaymentTime: time.Now(),UserID:      "1001",}// 通知观察者fmt.Println("=== 开始通知观察者 ===")orderService.Notify(event)fmt.Println("=== 所有通知处理完成 ===")
}

执行结果

=== RUN   Test_test
=== 开始通知观察者 ===
物流系统:订单20230303123456发货,用户1001,金额2999.00
[Info] LogisticsObserver处理完成
用户服务:订单20230303123456创建成功,用户1001,金额2999.00
[Info] UserObserver处理完成
库存系统:订单20230303123456扣减库存,用户1001,金额2999.00
[Info] InventoryObserver处理完成
=== 所有通知处理完成 ===
--- PASS: Test_test (0.00s)
PASS

五、高级应用

1. 异步通知机制(参考网页10)

func (w *WeatherStation) asyncNotify() {w.mu.Lock()observers := make([]Observer, len(w.observers))copy(observers, w.observers)w.mu.Unlock()var wg sync.WaitGroupfor _, o := range observers {wg.Add(1)go func(obs Observer) {defer wg.Done()obs.Update(w.temp)}(o)}wg.Wait()
}

2. 事件过滤机制

type SmartObserver struct {lastTemp float64
}func (s *SmartObserver) Update(temp float64) {if math.Abs(temp-s.lastTemp) > 1.0 {fmt.Println("温度变化超过1度")s.lastTemp = temp}
}

六、与其他模式对比

模式核心区别典型场景
中介者模式通过中介协调多对象交互复杂对象间通信
责任链模式请求沿链传递 vs 广播通知审批流程处理
发布订阅引入消息代理解耦更彻底()分布式系统消息传递

七、实现建议

  1. 线程安全:使用sync.Mutex保证并发安全()
  2. 批量通知:使用copy避免通知过程中列表变更(见示例)
  3. 性能优化:对于大量观察者采用批处理或异步通知()
  4. 防止泄漏:及时移除无效观察者()

八、典型应用

  1. GUI事件处理:如按钮点击事件监听()
  2. 微服务配置中心:配置变更通知服务集群()
  3. 游戏引擎:玩家状态变化触发多系统响应()

通过这种设计模式,可以构建出松耦合、易扩展的事件驱动系统。在实际Go开发中,结合channel特性还能实现更高效的观察者模式变体。

相关文章:

Golang学习笔记_41——观察者模式

Golang学习笔记_38——享元模式 Golang学习笔记_39——策略模式 Golang学习笔记_40——模版方法模式 文章目录 一、核心概念1. 定义2. 解决的问题3. 核心角色4. 类图 二、特点分析三、适用场景1. 股票价格监控系统2. 物联网设备状态监控3. 电商订单状态通知 四、Go语言实现示例…...

中原银行:从“小机+传统数据库”升级为“OceanBase+通用服务器”,30 +系统成功上线|OceanBase DB大咖说(十五)

OceanBase《DB 大咖说》第 15 期&#xff0c;我们邀请到了中原银行金融科技部数据团队负责人&#xff0c;吕春雷。本文为本期大咖说的精选。 吕春雷是一位资历深厚的数据库专家&#xff0c;从传统制造企业、IT企业、甲骨文公司到中原银行&#xff0c;他在数据库技术与运维管理…...

slam学习笔记9---ubuntu2004部署interactive_slam踩坑记录

背景&#xff1a;interactive_slam是一款可用于离线优化点云地图算法。部署安装容易出问题&#xff0c;这里记录一下。 一、安装基本流程 绝大部分跟着readme走&#xff0c;g2o安装使用apt安装 interactive_slam depends on the following libraries:GL3W GLFW Dear ImGui p…...

MVC模式全解析

MVC 模式&#xff1a;概念与架构基石 在软件开发的广袤宇宙中&#xff0c;MVC 模式宛如一颗璀璨的恒星&#xff0c;照亮了无数开发者前行的道路。它是一种经典的软件架构模式&#xff0c;全称为 Model - View - Controller&#xff0c;即模型 - 视图 - 控制器 &#xff0c;将应…...

游戏引擎学习第140天

回顾并为今天的内容做准备 目前代码的进展到了声音混音的部分。昨天我详细解释了声音的处理方式&#xff0c;声音在技术上是一个非常特别的存在&#xff0c;但在游戏中进行声音混音的需求其实相对简单明了&#xff0c;所以今天的任务应该不会太具挑战性。 今天我们会编写一个…...

LeetCode热题100JS(44/100)第八天|二叉树的直径|二叉树的层序遍历|将有序数组转换为二叉搜索树|验证二叉树搜索树|二叉搜索树中第K小的元素

543. 二叉树的直径 题目链接&#xff1a;543. 二叉树的直径 难度&#xff1a;简单 刷题状态&#xff1a;1刷 新知识&#xff1a; 解题过程 思考 示例 1&#xff1a; 输入&#xff1a;root [1,2,3,4,5] 输出&#xff1a;3 解释&#xff1a;3 &#xff0c;取路径 [4,2,1,3] 或…...

【虚拟化】Hyper-V 与 WSL 2

关于 Hyper-V 与 WSL 2 的简介 Hyper-V 是微软出的 Type-I 型 Hypervisor&#xff0c;根据微软官方说 WSL 2 用了 Hyper-V 架构的子集&#xff0c;称为虚拟机平台&#xff08;Virtual Machine Platform&#xff09;&#xff0c;是 Windows 中的一个可选组件&#xff0c;所以你…...

力扣刷题DAY6(滑动窗口/中等+栈/简单、中等)

一、滑动窗口 找到字符串中所有字母异位词 方法一&#xff1a;哈希表 class Solution { public:vector<int> findAnagrams(string s, string p) {vector<int> ans;unordered_map<char, int> target;for (int i 0; i < p.size(); i) {target[p[i]];}in…...

MySQL中的共享锁和排他锁

MySQL 中的锁可以从多个维度进行分类&#xff0c;其中从模式上可以分为共享锁&#xff08;Shared Lock&#xff0c;S Lock&#xff09;和 排他锁&#xff08;Exclusive Lock&#xff0c;X Lock&#xff09;。 共享锁&#xff08;Shared Lock&#xff0c;S Lock&#xff09; 共…...

HarmonyOS ArkTS声明式UI开发实战教程

引言&#xff1a;为何选择ArkTS&#xff1f; 在HarmonyOS生态快速发展的当下&#xff0c;ArkTS作为新一代声明式UI开发框架&#xff0c;正在引发移动应用开发范式的变革。笔者曾在多个跨平台框架开发中经历过"命令式编程之痛"&#xff0c;直到接触ArkTS后才发现&…...

【AI】【Unity】关于Unity接入DeepseekAPI遇到的坑

前言 由于deepseek网页端在白天日常抽风&#xff0c;无法正常的使用&#xff0c;所以调用API就成了目前最好的选择&#xff0c;尤其是Deepseek的API价格低得可怕&#xff0c;这不是和白送的一样吗&#xff01;然后使用过很多本地部署接入API的方式&#xff0c;例如Chatbox、Pa…...

虚拟机 | Ubuntu图形化系统: open-vm-tools安装失败以及实现文件拖放

系列文章目录 虚拟机 | Ubuntu 安装流程以及界面太小问题解决 文章目录 系列文章目录虚拟机 | Ubuntu 安装流程以及界面太小问题解决 前言一、VMware Tools 和 open-vm-tools 是什么1、VMware Tools2、open-vm-tools 二、推荐使用open-vm-tools&#xff08;简单&#xff09;1、…...

zabbix 监控的内存溢出问题 out of memory

zabbix 由于是docker 建置 zabbix-server-server 在web端出现异常情况 经过检查发现docker status 显示Restarting (0) 4 seconds ago 频繁在重起 经过docker logs 分析查看到 __zbx_shmem_malloc(): out of memory (requested 256 bytes) 问题 191:20250306:023902.993 m…...

【DeepSeek】Ubuntu快速部署DeepSeek(Ollama方式)

文章目录 人人都该学习的DeepSeekDeepSeek不同版本功能差异DeepSeek与硬件直接的关系DeepSeek系统兼容性部署方式选择部署步骤&#xff08;Ollama方式&#xff09;1.选定适合的deepseek版本2.环境准备3.安装Ollama4.部署deepseek5.测试使用 人人都该学习的DeepSeek DeepSeek 作…...

升级到Android Studio 2024.2.2 版本遇到的坑

一、上来就编译报错&#xff0c;大概率是因为选择了替换安装&#xff0c;本地配置文件出错 找到本地当前版本的配置文件&#xff0c;删掉&#xff0c;重启studio就好了&#xff1a; 1、打开终端 2、“cd /Users/用户名/Library/Application\ Support/Google” //到Google目录 …...

2025最新Transformer模型及深度学习前沿技术应用

第一章、注意力&#xff08;Attention&#xff09;机制 1、注意力机制的背景和动机&#xff08;为什么需要注意力机制&#xff1f;注意力机制的起源和发展里程碑&#xff09;。 2、注意力机制的基本原理&#xff08;什么是注意力机制&#xff1f;注意力机制的数学表达与基本公…...

深入解析 Umi-OCR:高效的免费开源 OCR 文字识别工具

1. Umi-OCR 简介 1.1 什么是 Umi-OCR&#xff1f; Umi-OCR 是一款开源、免费、支持离线使用的光学字符识别&#xff08;OCR&#xff09;工具&#xff0c;基于 PaddleOCR 和 Tesseract-OCR&#xff0c;能够高效识别图片中的文字&#xff0c;尤其适用于批量截图文字提取、PDF 文…...

WebAssembly技术及应用了解

WebAssembly&#xff08;Wasm&#xff09;是一种为Web设计的高效、低级的二进制指令格式&#xff0c;旨在提升Web应用的性能并支持多种编程语言。以下是对其核心概念、优势、应用场景及开发流程的系统介绍&#xff1a; 1. 核心概念 二进制格式&#xff1a;Wasm采用紧凑的二进制…...

【ESP-ADF】在 VSCode 安装 ESP-ADF 注意事项

1.检查网络 如果您在中国大陆安装&#xff0c;请使用魔法上网&#xff0c;避免无法 clone ESP-ADF 仓库。 2.VSCode 安装 ESP-ADF 在 VSCode 左侧活动栏选择 ESP-IDF:explorer&#xff0c;展开 advanced 并点击 Install ESP-ADF 然后会出现选择 ESP-ADF 安装目录。 如果出现…...

我的两个医学数据分析技术思路

我的两个医学数据分析技术思路 从临床上获得的或者公共数据库数据这种属于观察性研究&#xff0c;是对临床诊疗过程中自然产生的数据进行分析而获得疾病发生发展的规律等研究成果。再细分&#xff0c;可以分为独立危险因素鉴定和预测模型构建两种。 独立危险因素鉴定是一直以…...

MySQL 架构、索引优化、DDL解析、死锁排查

私人博客传送门 MySQL 认识索引 | 魔筝炼药师 MySQL 索引优化 | 魔筝炼药师 OnlineDDL&#xff08;在 MySQL 5.7 数据库里&#xff0c;InnoDB引擎&#xff0c;执行一条DDL会发生什么事情&#xff09; | 魔筝炼药师 MySQL 死锁排查 | 魔筝炼药师...

Ubuntu搭建最简单WEB服务器

安装apache2 sudo apt install apache2 检查状态 $ sudo systemctl status apache2 ● apache2.service - The Apache HTTP ServerLoaded: loaded (/lib/systemd/system/apache2.service; enabled; vendor prese>Active: active (running) since Thu 2025-03-06 09:51:10…...

VSCode 配置优化指南:打造极致高效的前端开发环境

VSCode 配置优化指南&#xff1a;打造极致高效的前端开发环境 一、基础环境配置&#xff1a;让开发更流畅 1. 性能优化设置 // settings.json {"files.autoSave": "afterDelay", // 自动保存&#xff08;延迟1秒&#xff09;"files.exclud…...

FPGA-DE2115开发板实现4位全加器、3-8译码器。

文章目录 一、安装quartus二、4位全加器三、3-8译码器&#xff08;8段数码管&#xff09;四、参考文章 一、安装quartus 安装quartus参考文章&#xff1a;Quartus Prime 18.0与ModelSim的安装 Quartus II 18.0安装教程&#xff08;非常详细&#xff09;从零基础入门到精通&…...

宇树科技嵌入式面试题及参考答案(春晚机器人的公司)

目录 设计一个带看门狗(Watchdog)的嵌入式系统,描述故障恢复流程 在资源受限的 MCU 上实现 OTA 升级功能,描述关键设计点 如何实现 OTA(空中升级)功能?描述固件校验和回滚机制的设计要点 推挽输出与开漏输出的区别?举例说明其在 GPIO 控制中的应用 UART、SPI、I2C …...

Spring提供的SPEL表达式

SPEL 1. 概述 SpEL是Spring框架中用于表达式语言的一种方式。它类似于其他编程语言中的表达式语言&#xff0c;用于在运行时计算值或执行特定任务。 SpEL提供了一种简单且强大的方式来访问和操作对象的属性、调用对象的方法&#xff0c;以及实现运算、条件判断等操作。它可以…...

Pytorch xpu环境配置 Pytorch使用Intel集成显卡

1、硬件集显要为Intel ARC并安装正确驱动 2、安装Intel oneAPI Base Toolkit &#xff08;https://www.intel.cn/content/www/cn/zh/developer/tools/oneapi/base-toolkit-download.html&#xff09;安装后大约20G左右&#xff0c;注意安装路径 3、安装Visual Studio Build To…...

uploadlabs通关思路

目录 靶场准备 复现 pass-01 代码审计 执行逻辑 文件上传 方法一&#xff1a;直接修改或删除js脚本 方法二&#xff1a;修改文件后缀 pass-02 代码审计 文件上传 1. 思路 2. 实操 pass-03 代码审计 过程&#xff1a; 文件上传 pass-04 代码审计 文件上传 p…...

优选算法的智慧之光:滑动窗口专题(二)

专栏&#xff1a;算法的魔法世界​​​​​​ 个人主页&#xff1a;手握风云 目录 一、例题讲解 1.1. 最大连续1的个数 III 1.2. 找到字符串中所有字母异位词 1.3. 串联所有单词的子串 1.4. 最小覆盖子串 一、例题讲解 1.1. 最大连续1的个数 III 题目要求是二进制数组&am…...

【蓝桥杯单片机】第十二届省赛

一、真题 二、模块构建 1.编写初始化函数(init.c) void Cls_Peripheral(void); 关闭led led对应的锁存器由Y4C控制关闭蜂鸣器和继电器 由Y5C控制 2.编写LED函数&#xff08;led.c&#xff09; void Led_Disp(unsigned char ucLed); 将ucLed取反的值赋给P0 开启锁存器…...