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

Go设计模式-观察者模式

简介

在软件开发中,我们常常会遇到这样的场景:一个对象的状态变化需要通知到多个其他对象,让它们做出相应的反应。观察者模式(Observer Pattern)就是解决这类问题的一种设计模式。在 Go 语言中,由于其简洁高效的特性,实现观察者模式也有独特的方式。本文将深入探讨 Golang 中观察者模式的基础概念、使用方法、常见实践以及最佳实践,帮助读者更好地掌握并运用这一模式。

基础概念

什么是观察者模式

观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听一个主题对象。这个主题对象在状态发生变化时,会通知所有观察者对象,使它们能够自动更新自己的状态。

角色与职责

主题(Subject):也称为被观察对象,它维护一个观察者列表,并提供注册、注销和通知观察者的方法。当主题的状态发生变化时,会调用通知方法,遍历观察者列表并调用每个观察者的更新方法。
观察者(Observer):定义一个更新接口,当主题状态发生变化时,主题会调用这个接口通知观察者。观察者实现这个接口,在接口方法中定义自己的更新逻辑。

使用方法

定义观察者接口

在 Go 语言中,我们可以通过接口来定义观察者的行为。

// Observer 接口定义了观察者的更新方法
type Observer interface {Update(message string)
}

定义被观察对象

被观察对象需要维护一个观察者列表,并提供注册、注销和通知方法。

// Subject 结构体表示被观察对象
type Subject struct {observers []Observerstate     string
}// Register 方法用于注册观察者
func (s *Subject) Register(o Observer) {s.observers = append(s.observers, o)
}// Unregister 方法用于注销观察者
func (s *Subject) Unregister(o Observer) {for i, observer := range s.observers {if observer == o {s.observers = append(s.observers[:i], s.observers[i+1:]...)break}}
}// Notify 方法用于通知所有观察者
func (s *Subject) Notify() {for _, observer := range s.observers {observer.Update(s.state)}
}// SetState 方法用于设置被观察对象的状态
func (s *Subject) SetState(state string) {s.state = states.Notify()
}

完整示例

package mainimport "fmt"// Observer 接口定义了观察者的更新方法
type Observer interface {Update(message string)
}// Subject 结构体表示被观察对象
type Subject struct {observers []Observerstate     string
}// Register 方法用于注册观察者
func (s *Subject) Register(o Observer) {s.observers = append(s.observers, o)
}// Unregister 方法用于注销观察者
func (s *Subject) Unregister(o Observer) {for i, observer := range s.observers {if observer == o {s.observers = append(s.observers[:i], s.observers[i+1:]...)break}}
}// Notify 方法用于通知所有观察者
func (s *Subject) Notify() {for _, observer := range s.observers {observer.Update(s.state)}
}// SetState 方法用于设置被观察对象的状态
func (s *Subject) SetState(state string) {s.state = states.Notify()
}// ConcreteObserver 结构体实现了 Observer 接口
type ConcreteObserver struct {name string
}// Update 方法实现了观察者的更新逻辑
func (co *ConcreteObserver) Update(message string) {fmt.Printf("%s 接收到更新: %s\n", co.name, message)
}func main() {subject := &Subject{}observer1 := &ConcreteObserver{name: "观察者1"}observer2 := &ConcreteObserver{name: "观察者2"}subject.Register(observer1)subject.Register(observer2)subject.SetState("新状态")subject.Unregister(observer2)subject.SetState("又一个新状态")
}

在这个示例中,我们定义了 Observer 接口和 Subject 结构体。ConcreteObserver 结构体实现了 Observer 接口的 Update 方法。在 main 函数中,我们创建了一个 Subject 对象和两个 ConcreteObserver 对象,并进行了注册、状态设置和注销等操作。

常见实践

事件驱动的系统

在事件驱动的系统中,观察者模式非常有用。例如,在一个图形用户界面(GUI)应用中,按钮的点击事件可以被视为一个主题,而各个需要对点击事件做出反应的组件(如文本框更新、菜单显示等)可以被视为观察者。当按钮被点击(主题状态变化)时,所有注册的观察者会收到通知并执行相应的操作。

状态管理

在状态管理场景中,一个对象的状态变化可能会影响到多个其他对象。比如,在一个游戏中,角色的状态(如生命值、等级等)变化时,可能需要通知游戏界面更新显示、道具系统调整道具可用性等。通过观察者模式,可以方便地实现这种一对多的状态变化通知。

最佳实践

并发安全

在多线程环境下使用观察者模式时,需要注意并发安全。可以使用 Go 语言的 sync 包来实现线程安全的注册、注销和通知操作。例如,在 Subject 结构体中添加一个互斥锁:

// Subject 结构体表示被观察对象
type Subject struct {observers []Observerstate     stringmu        sync.Mutex
}// Register 方法用于注册观察者
func (s *Subject) Register(o Observer) {s.mu.Lock()s.observers = append(s.observers, o)s.mu.Unlock()
}// Unregister 方法用于注销观察者
func (s *Subject) Unregister(o Observer) {s.mu.Lock()for i, observer := range s.observers {if observer == o {s.observers = append(s.observers[:i], s.observers[i+1:]...)break}}s.mu.Unlock()
}// Notify 方法用于通知所有观察者
func (s *Subject) Notify() {s.mu.Lock()observersCopy := make([]Observer, len(s.observers))copy(observersCopy, s.observers)s.mu.Unlock()for _, observer := range observersCopy {observer.Update(s.state)}
}

内存管理

在注销观察者时,要确保正确地从观察者列表中移除,避免内存泄漏。同时,在观察者的实现中,要注意避免循环引用,防止对象无法被垃圾回收。

接口设计

观察者接口的设计要尽可能简洁和通用,只包含必要的方法。这样可以提高代码的可维护性和扩展性。如果需要传递更复杂的信息,可以考虑将相关数据封装在一个结构体中,作为 Update 方法的参数。

小结

观察者模式是一种强大的设计模式,在 Golang 中通过接口和结构体的组合可以方便地实现。理解其基础概念、掌握使用方法,并遵循最佳实践,能够帮助我们在开发中更好地处理对象之间的依赖关系,提高代码的可维护性和扩展性。无论是在事件驱动的系统还是状态管理等场景中,观察者模式都能发挥重要作用。希望本文能帮助读者深入理解并灵活运用 Golang 观察者模式。

相关文章:

Go设计模式-观察者模式

简介 在软件开发中,我们常常会遇到这样的场景:一个对象的状态变化需要通知到多个其他对象,让它们做出相应的反应。观察者模式(Observer Pattern)就是解决这类问题的一种设计模式。在 Go 语言中,由于其简洁…...

《TCP/IP详解 卷1:协议》之第七、八章:Ping Traceroute

目录 一、ICMP回显请求和回显应答 1、ICMP回显请求 2、ICMP回显应答 二、ARP高速缓存 三、IP记录路由选项(Record Route,RR) 1、记录路由选项的工作过程 2、RR 选项的 IP 头部格式 2.1、RR 请求 2.2、RR响应 四、ping 的去返路径 五…...

Unity任务系统笔记

数据结构设计 任务基类包括的字段: string 任务内容; Transform 任务目的地; MyCharacter 任务开启后要更新对话的NPC; MyTalkData 任务开启后相关NPC要说的对话数据; 共同方法:开启任务、完成任务。…...

Three.js + React 实战系列-3D 个人主页:构建 Hero 场景组件(项目核心)✨

在本节中,我们将完成整个 3D 主业项目中最核心的组件 —— Hero.jsx。 这个组件作为首页的主视觉部分,整合了 3D 模型、动画相机、交互按钮与自适应布局,构建出一个立体、酷炫、可交互的主场景。 前置准备: ✅安装依赖&#xff…...

线程池(二):深入剖析synchronized关键字的底层原理

线程池(二):深入剖析synchronized关键字的底层原理 线程池(二):深入剖析synchronized关键字的底层原理一、基本使用1.1 修饰实例方法1.2 修饰静态方法1.3 修饰代码块 二、Monitor2.1 Monitor的概念2.2 Moni…...

网络原理 - 9

目录 数据链路层 以太网 以太网帧格式 MAC 地址 DNS(Domain Name System) 完! 数据链路层 这里的内容也是简单了解,除非是做交换机开发,一般程序员不需要涉及~~ 以太网 ”以太网“不是一种具体的网络&#xf…...

springboot入门-业务逻辑核心service层

在 Spring Boot 中,Service 层是业务逻辑的核心,负责协调数据访问层(Repository 或 Mapper)和控制器层(Controller),处理业务规则、事务管理以及数据转换。以下是 Service 层的详细说明、常用注…...

在RHEL 10上安装和配置TFTP服务器(不使用xinetd)

RHEL10已经废弃xinetd,使用下面的方式安装配置TFTP服务器。 1. 安装TFTP服务器和客户端 sudo dnf install tftp-server tftp -y 2. 配置TFTP服务器 创建TFTP根目录并设置权限 sudo mkdir -p /var/lib/tftpboot sudo chmod -R 777 /var/lib/tftpboot sudo chown -R…...

AIGC在游戏开发中的革命:自动化生成3A级游戏内容

一、智能游戏开发架构 1.1 传统开发痛点与AIGC创新 开发环节 传统痛点 AIGC解决方案 角色原画设计 美术资源产能瓶颈 文生图3D模型自动生成 场景搭建 重复劳动占比高 程序化生成风格迁移 NPC行为设计 模式化严重 强化学习驱动智能行为 任务系统 剧情线性缺乏变化 动态剧情生成系…...

ChatGPT、deepseek、豆包、Kimi、通义千问、腾讯元宝、文心一言、智谱清言代码能力对比

均使用测试时的最强模型 均是一次对话,对话内容一样 均开启深度思考 能联网的都联网了,但是作用不大,因为蓝桥杯刚考完,洛谷题目刚上传没多久 问题一测试了两遍 从问题三开始不再测试智谱清言(它思考时间太长了,前两个…...

Linux扩展

目录 扩展 查找如何进行后台运行程序的指令 使用 & 符号 使用 nohup 命令 使用 screen 或 tmux find命令 基本语法 常用选项 grep 命令 基本语法 常用选项 如何使用 vim 直接定位到错误行 1. 使用 :make 和 :copen 2. 使用 :lineno 定位 3. 通过 :grep 或 :…...

Java Hotspot VM researcher

** therefore, careful design and understanding of modules are essential to fully reap the performance benefits. **...

java—基础

目标 ├── 第一阶段:内容清单 │ └── 目标:建立编程思想 ├── 第二阶段:内容清单 │ └── 目标:提升编程能力 └── 第三阶段:内容清单└── 目标:分析需求,代码实现能力以下是根…...

【OpenCV】第二章——图像处理基础

图像处理基础学习笔记 本章节详细介绍了图像处理的基础内容,包括图像的读取、显示、保存,基本属性的查看,图像的变换与操作,以及常用的图像处理方法。 目录 图像的读取与显示图像基本属性图像的灰度化与二值化图像的色彩空间转换…...

在WSL2+Ubuntu22.04中通过conda pack导出一个conda环境包,然后尝试导入该环境包

如何导出一个离线conda环境?有两种方式,一种是导出env.yml即环境配置,一种是通过conda pack导出为一个环境包,前者只是导出配置(包括包名、版本等),而后者是直接将环境中所有的内容打包&#xf…...

C++:类和对象(上)---镜中万象:C++类的抽象之境与对象的具体之象

类(Class)是一种用户自定义的数据类型。 文章目录: 前言一、面向过程和面向对象初步认识 二、类的引入 三、类的定义 3.1类是什么? 3.2类的定义 四、类的访问限定符和封装 4.1类的访问限定符 4.2封装 五、类和对象的关系 六、类对…...

碰一碰发视频源码搭建全解析,支持OEM

在数字化交互体验不断升级的背景下,“碰一碰发视频” 功能凭借其便捷性和趣味性,逐渐成为营销推广、社交分享等场景中的热门需求。该功能基于近场通信技术,实现设备间快速的数据传输。本文将详细介绍其源码搭建过程,助力开发者实现…...

搭建spark-local模式

要搭建Spark的local模式,你可以按照以下步骤进行操作(以在Linux系统上安装为例,假设你已经安装了Java环境): 1. 下载Spark安装包:访问Spark官方网站(https://spark.apache.org/downloads.html&a…...

Goland终端PowerShell命令失效

Goland终端Terminal的PowerShell不能使用,明明windows上升级了PowerShell 7设置了配置文件,但是只能在windows终端下使用,goland终端下直接失效报错,安装升级PowerShell请看[博客](Windows11终端升级PowerShell7 - HashFlag - 博客…...

前端节流、防抖函数

节流 什么是节流? 节流就是同一个事件 一秒钟他执行了很多次。但是我不想他执行这么多次,我只想让他执行一次 或者两次。 那该怎么办? why baby why 那我想就是他执行的时候 我就设置一个定时器,如果定时器是空的,等会…...

如何使用WebRTC

WebRTC比较容易使用,只需要很少的步骤,有些消息在浏览器和服务器之间流动,有些则直接在两个浏览器之间流动, 1、建立WebRTC会话 a:建立WebRTC连接需要加入以下几个步骤: 获取本地媒体:getUse…...

在 Vue 3 setup() 函数中使用 TypeScript 处理 null 和 undefined 的最佳实践

在 Vue 3 中使用 setup() 函数和 TypeScript 时,null 和 undefined 是两个需要特别关注的类型。虽然它们看起来都表示“没有值”,但它们在 JavaScript 和 TypeScript 中有着不同的含义和使用场景。如果不小心处理它们,可能会导致潜在的 bug 或…...

【C++11】Lambda表达式

前言 上文我们学习了C11新语法,可变参数模板以及用可变参数模板作为形参的emplace接口。【C11】可变参数模板-CSDN博客 本文我们来学习C11下一个新语法,Lambda表达式。 1.Lambda表达式语法 Lambda表达式本质是一个匿名函数对象,与普通函数不同…...

【落羽的落羽 C++】vector

文章目录 一、vector类介绍二、vector中的常用接口三、迭代器失效问题四、vector的使用实例五、vector模拟实现 一、vector类介绍 vector是STL中的一种容器,本质上是顺序表。它和string类的结构很相似,其也有size、capacity、数组等,不同的是…...

DIFY 浅尝 - Dify + Ollama 抓取BBC新闻

假设你已经按照上篇文章 DIFY 浅尝 - DIFY Ollama 添加模型搭建好了本地环境. 创建一个新的工作流 进入你的本地Dify工作台,选择工作室->创建空白应用 选择工作流,输入应用名称BBC旅游新闻,点击创建 创建一个网页爬虫 配置网页爬虫…...

基于MTF的1D-2D-CNN-BiLSTM-Attention时序图像多模态融合的故障分类识别(Matlab完整源码和数据),适合研究学习,附模型研究报告

基于MTF的1D-2D-CNN-BiLSTM-Attention时序图像多模态融合的故障分类识别(Matlab完整源码和数据),适合研究学习,附模型研究报告 目录 基于MTF的1D-2D-CNN-BiLSTM-Attention时序图像多模态融合的故障分类识别(Matlab完整…...

nuxt3项目搭建:一、初始化项目流程指南

一、初始化项目 初始化命令 1、创建nuxt3项目 npm create nuxtlatest2、填写项目名称 这里我直接填了nuxt-app 3、选择包管理器 这里的包管理器我们选择pnpm 4、选择是否创建git仓库 选择完包管理器后,脚手架会自动下载依赖,git仓库我已经创建好了…...

案例速成GO+redis 个人笔记

更多个人笔记:(仅供参考,非盈利) gitee: https://gitee.com/harryhack/it_note github: https://github.com/ZHLOVEYY/IT_note (更多GOredis等见内部,会及时更新~&#x…...

C/C++ 头文件包含机制:从语法到最佳实践

在C/C++编程中,头文件(.h 或 .hpp)扮演着至关重要的角色。它们不仅是代码模块化的基石,更是编译器理解程序结构的关键。然而,头文件的使用看似简单,实则暗含许多细节,稍有不慎便可能导致编译错误、代码冗余,甚至隐藏难以调试的问题。本文将从语法、编译器行为到工程实践…...

职业教育新形态数字教材的建设与应用:重构教育生态的数字化革命

教育部新时代职业学校名师(名匠)名校长培养计划、四川省第四批职业学校名师(名匠)培养计划专题 在某职业院校的智能制造课堂上,学生佩戴VR设备,通过数字教材中的虚拟工厂完成设备装配训练,系统实时生成操作评分与改进建议。这一场景折射出职业…...

跟着deepseek学golang--Go vs Java vs JavaScript三语言的差异

文章目录 一、类型系统与编译方式1. 类型检查时机2. 空值安全设计 二、并发模型对比1. 并发单元实现4. 锁机制差异 三、内存管理机制1. 垃圾回收对比2. 对象模型差异 四、工程实践差异1. 依赖管理工具4. 异常处理范式 五、跨平台能力对比1. 编译输出目标 综合对比表​​五角星说…...

梯度下降法

梯度下降法是一种常见的求最小值(或最值)的方法。它是通过沿着函数梯度的负方向进行迭代更新,直到找到局部最小值或最大值。梯度下降法应用于多元函数时,通过更新参数的方式找到最优解。 梯度下降法步骤: 初始化参数&…...

【Java 数据结构】List,ArrayList与顺序表

目录 一. List 1.1 什么是List 1.2 List 的常见方法 1.3 List 的使用 二. 顺序表 2.1 什么是顺序表 2.2 实现自己的顺序表 2.2.1 接口实现 2.2.2 实现顺序表 三. ArrayList 3.1 ArrayList简介 3.2 ArrayList的三个构造方法 3.2.1 无参构造方法 3.2.2 带一个参数的…...

用Python做有趣的AI项目1:用 TensorFlow 实现图像分类(识别猫、狗、汽车等)

项目目标 通过构建卷积神经网络(CNN),让模型学会识别图片中是什么物体。我们将使用 CIFAR-10 数据集,它包含 10 类:飞机、汽车、鸟、猫、鹿、狗、青蛙、马、船和卡车。 🛠️ 开发环境与依赖 安装依赖&…...

正确应对监管部门的数据安全审查

首席数据官高鹏律师团队编著 在当今数字化时代,数据安全已成为企业及各类组织面临的重要议题,而监管部门的数据安全审查更是关乎其生存与发展的关键挑战。随着法律法规的不断完善与监管力度的加强,如何妥善应对这一审查,避免潜在…...

Springboot用IDEA打jar包 运行时 错误: 找不到或无法加载主类

Springboot用IDEA打jar包 运行时 错误: 找不到或无法加载主类 今天遇到个很神奇的问题。 就是我在打包我项目后。用java -jar命令的话 是无法启动这个项目的。 但是我在idea里面进行运行 就可以运行 先说结论 因为我这个是jdk17的项目 而我本机的jdk是1.8 所以说就会出现…...

【Linux网络】构建与优化HTTP请求处理 - HttpRequest从理解到实现

📢博客主页:https://blog.csdn.net/2301_779549673 📢博客仓库:https://gitee.com/JohnKingW/linux_test/tree/master/lesson 📢欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正! &…...

【高频考点精讲】实现垂直居中的多种CSS方法比较与最佳实践

前端工程师必看:7种CSS垂直居中方案大比拼(附真实代码) 今天咱们聊聊前端开发中最让人头疼的问题之一——垂直居中。不知道你们有没有遇到过这种情况:明明设置了margin: 0 auto水平居中了,垂直方向怎么折腾都不对劲。全栈老李当年刚入行时,为了一个div居中能折腾一晚上,…...

Java 字符串基础介绍

在 Java 编程中,字符串是不可或缺的一部分。无论是用户界面的消息显示、文件路径的处理,还是用户信息的存储,字符串都扮演着至关重要的角色。本文将带您深入了解 Java 字符串的特性、用法以及一些高级技巧,帮助您在编程实践中更加…...

SpringBoot中暗藏的设计模式

一、工厂模式 想象一下你去奶茶店点单——你只需要告诉店员要"珍珠奶茶",后厨就会自动完成煮茶、加料、封口整套流程。这就是工厂模式在SpringBoot中的体现。 典型应用场景: Bean的创建过程(ApplicationContext就是超级工厂&…...

使用 AFL++ 对 IoT 二进制文件进行模糊测试 - 第一部分

American fuzzy lop 是一款面向安全的模糊测试器,它采用一种新型的编译时插桩和遗传算法,可以自动发现干净、有趣的测试用例,从而触发目标二进制文件中新的内部状态。这显著提高了模糊测试代码的功能覆盖率。 AFL 的地址是https://lcamt uf.coredump.cx/afl/ 。它已经有一段…...

Java 线程的六种状态与完整生命周期详解

🚀 Java 线程的几种状态详解 在 Java 中,线程状态(Thread State)是由 Thread.State 枚举定义的,总共有六种: 状态含义典型场景示例NEW新建状态,线程对象刚创建,还未调用 start() 方…...

常见的机器视觉通用软件

国际常用软件 OpenCV : 特点 :开源免费,社区支持强大,拥有丰富的图像处理和计算机视觉算法库,支持多种编程语言,如 C、Python、Java 等,可实现对象检测、图像分割、特征提取等功能,具…...

使用 Frida 绕过 iOS 应用程序中的越狱检测

在这篇博文中,我们将介绍**Frida**,它是用于移动应用程序安全分析的真正有趣的工具之一。 我们在高级 Android 和 iOS 漏洞利用培训中也深入讲解了这一点,您可以在这里注册 -培训链接 即使您从未使用过 Frida,本文也将作为指南,帮助您进入 Frida 的世界,进行移动应用程…...

创建可执行 JAR 文件

📦 创建可执行 JAR 文件 🔹 概述 在完成示例的最后环节,我们将创建一个完全自包含的可执行 jar 文件,该文件可直接在生产环境运行。可执行 jar(又称 uber jar 或 fat jar)是一种包含编译后类文件及全部运行…...

LIDC-IDRI数据集切割代码教程【pylidc库】

数据集: 通过网盘分享的文件:LIDC 链接: 百度网盘 请输入提取码 提取码: ywb8 代码: 通过网盘分享的文件:LIDC-IDRI-Preprocessing.rar 链接: 百度网盘 请输入提取码 提取码: b1za 【代码里的部分数据就不删了,方…...

Java数据结构——Stack

Stack 栈的概念和使用栈的概念栈的使用 栈的应用出栈元素序列有效的括号栈的压入、弹出序列逆波兰表达式最小栈 栈的概念和使用 栈的概念 栈(Stack):一种特殊的线性表,只允许再栈的一端进行插入和删除元素,这一端点被称为栈顶,另…...

SMT贴片加工费控制与优化实践指南

内容概要 SMT贴片加工费的控制与优化需建立在对成本结构的系统性认知基础上。本节从物料采购、设备运行、工艺参数三大维度切入,结合BOM清单管理、钢网使用规范等实操环节,构建覆盖全流程的降本增效框架。以下表格列举了SMT加工成本的典型构成要素及其占…...

Eclipse 插件开发 4 工具栏

Eclipse 插件开发 4 工具栏 1 增加工具(push)2 增加工具(toggle)3 增加工具(radio) 位置locationURI备注菜单栏menu:org.eclipse.ui.main.menu添加到传统菜单工具栏toolbar:org.eclipse.ui.main.toolbar添加到工具栏 style 值含义显示效果push普通按钮(默认&#x…...

Dify中的文本分词处理技术详解

Dify中的文本分词处理技术详解 引言核心架构概览索引处理器工厂 文本分词技术详解基础分词器增强型递归字符分词器固定分隔符文本分词器递归分割算法 索引处理器中的分词应用特殊索引处理器的分词特点问答索引处理器父子索引处理器 分词技术的应用场景技术亮点与优势总结 引言 …...