首页 > 后端开发 > Golang > 正文

Golang观察者模式实现 channel事件通知机制

P粉602998670
发布: 2025-08-20 08:52:01
原创
909人浏览过
Go中观察者模式可通过channel和goroutine实现,核心为Subject维护观察者channel列表,状态变更时通过Notify向各observer广播Event,利用RWMutex保证并发安全,select+default实现非阻塞通知,Close方法关闭所有channel防止泄漏,适用于事件驱动架构。

golang观察者模式实现 channel事件通知机制

在Go语言中,观察者模式可以通过

channel
登录后复制
goroutine
登录后复制
天然地实现事件通知机制。这种模式适用于解耦事件发布者和多个订阅者,比如在系统状态变更、消息广播或事件驱动架构中非常实用。

观察者模式核心思想

观察者模式定义了一种一对多的依赖关系,当一个对象(被观察者)的状态发生变化时,所有依赖它的对象(观察者)都会收到通知并自动更新。

在Go中,我们可以使用

channel
登录后复制
作为消息传递的媒介,利用其并发安全的特性,避免显式加锁。

基本结构设计

我们需要三个核心组件:

立即学习go语言免费学习笔记(深入)”;

  • Event:表示事件数据
  • Observer:接收事件的订阅者
  • Subject:事件发布者,维护观察者列表并通过channel广播

以下是一个简洁的实现示例:

package main

import (
    "fmt"
    "sync"
)

// Event 事件结构
type Event struct {
    Type string
    Data interface{}
}

// Observer 观察者接口
type Observer interface {
    Update(event Event)
}

// Subject 被观察者
type Subject struct {
    observers []chan Event // 所有观察者的channel
    mutex     sync.RWMutex
    closed    bool
}

func NewSubject() *Subject {
    return &Subject{
        observers: make([]chan Event, 0),
        closed:    false,
    }
}

// Subscribe 添加观察者,返回用于接收事件的channel
func (s *Subject) Subscribe() <-chan Event {
    s.mutex.Lock()
    defer s.mutex.Unlock()

    if s.closed {
        return nil
    }

    ch := make(chan Event, 10) // 带缓冲避免阻塞
    s.observers = append(s.observers, ch)
    return ch
}

// Notify 通知所有观察者
func (s *Subject) Notify(event Event) {
    s.mutex.RLock()
    defer s.mutex.RUnlock()

    for _, ch := range s.observers {
        select {
        case ch <- event:
        default:
            // 非阻塞发送,观察者处理慢也不会影响发布者
        }
        }
    }
}

// Close 关闭所有观察者channel
func (s *Subject) Close() {
    s.mutex.Lock()
    defer s.mutex.Unlock()

    if s.closed {
        return
    }
    s.closed = true

    for _, ch := range s.observers {
        close(ch)
    }
    s.observers = nil
}
登录后复制

使用示例

下面演示如何使用上述实现:

知我AI·PC客户端
知我AI·PC客户端

离线运行 AI 大模型,构建你的私有个人知识库,对话式提取文件知识,保证个人文件数据安全

知我AI·PC客户端 0
查看详情 知我AI·PC客户端
func main() {
    subject := NewSubject()

    // 创建两个观察者
    observer1 := subject.Subscribe()
    observer2 := subject.Subscribe()

    // 启动两个监听goroutine
    go func() {
        for event := range observer1 {
            fmt.Printf("Observer1 收到事件: %s, 数据: %v\n", event.Type, event.Data)
        }
        fmt.Println("Observer1 已关闭")
    }()

    go func() {
        for event := range observer2 {
            fmt.Printf("Observer2 收到事件: %s, 数据: %v\n", event.Type, event.Data)
        }
        fmt.Println("Observer2 已关闭")
    }()

    // 发布事件
    subject.Notify(Event{Type: "user.login", Data: "alice"})
    subject.Notify(Event{Type: "system.update", Data: "v1.0.0"})

    // 模拟运行一段时间后关闭
    subject.Close()

    // 防止主goroutine退出过早
    select {}
}
登录后复制

关键设计点说明

并发安全:使用

sync.RWMutex
登录后复制
保护观察者列表的读写操作,确保多goroutine环境下安全。

非阻塞通知:通过

select...default
登录后复制
实现非阻塞发送,避免某个观察者处理缓慢拖慢整个系统。

资源释放:提供

Close()
登录后复制
方法关闭所有channel,防止goroutine泄漏。

缓冲channel:每个观察者使用带缓冲的channel,提升异步处理能力。

基本上就这些。这种基于channel的实现简洁、高效,符合Go的并发哲学,适合大多数事件通知场景。

以上就是Golang观察者模式实现 channel事件通知机制的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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