设计模式大白话——观察者模式

文章目录

一、概述

​ 与其叫他观察者模式,我更愿意叫他叫 订阅-发布模式 ,这种模式在我们生活中非常常见,比如:追番了某个电视剧,当电视剧有更新的时候会第一时间通知你。当你预约了某款游戏后,游戏发布后就会立刻通知你,而不需要你每天都去关注游戏是否上线。

二、示例

​ 我想要说明一点,对于每一种设计模式,最重要的不是代码的结构,而是其中所蕴含的思想,你应该要去考虑的是代码为什么这么设计 ,而不是仅仅照着这样设计,因为每一种设计模式到不同的人手里写出来的代码多多少少会有些许差别,但是其核心的思想是一样的,那就是设计原则

  • 分析

    此模式需要发布者 及时地通知所有订阅者 ,因此发布者知道所有的订阅者的信息。发布者可以添加或者移除订阅者,订阅者也可以进行订阅或者取消订阅的操作。

  • 代码

    go 复制代码
    package main
    
    // Publisher 发布者接口
    type Publisher interface {
    	AddSubscriber(subscriber Subscriber)    // 添加订阅者
    	RemoveSubscriber(subscriber Subscriber) // 移除订阅者
    	NotifySubscribers()                     // 通知订阅者
    }
    
    // Subscriber 订阅者接口
    type Subscriber interface {
    	GetID() string                // 获取订阅者ID
    	Notify(contextMessage string) // 通知订阅者
    }
    
    // Object 被观察者, 实现 Publisher 接口
    type Object struct {
    	Subscribers    []Subscriber // 订阅者列表
    	ContextMessage string       // 上下文信息
    }
    
    func (p *Object) AddSubscriber(s Subscriber) {
    	p.Subscribers = append(p.Subscribers, s)
    }
    
    func (p *Object) RemoveSubscriber(s Subscriber) {
    	for i, subscriber := range p.Subscribers {
    		if subscriber.GetID() == s.GetID() {
    			p.Subscribers = append(p.Subscribers[:i], p.Subscribers[i+1:]...)
    		}
    	}
    }
    
    func (p *Object) NotifySubscribers() {
    	for _, subscriber := range p.Subscribers {
    		subscriber.Notify(p.ContextMessage)
    	}
    }
    
    // UpdateContextMessage 更新上下文信息
    func (p *Object) UpdateContextMessage(newMessage string) {
    	p.ContextMessage = newMessage
    	p.NotifySubscribers()
    }
    
    // SubscriberA 订阅者A, 实现 Subscriber 接口
    type SubscriberA struct {
    	ID string
    }
    
    func (s *SubscriberA) GetID() string {
    	return s.ID
    }
    
    func (s *SubscriberA) Notify(contextMessage string) {
    	println("SubscriberA received:", contextMessage)
    }
    
    // SubscriberB 订阅者B, 实现 Subscriber 接口
    type SubscriberB struct {
    	ID string
    }
    
    func (s *SubscriberB) GetID() string {
    	return s.ID
    }
    
    func (s *SubscriberB) Notify(contextMessage string) {
    	println("SubscriberB received:", contextMessage)
    }
    
    func main() {
    	// 创建被观察者
    	object := Object{}
    
    	// 创建订阅者
    	subscriberA := SubscriberA{ID: "subscriberA"}
    	subscriberB := SubscriberB{ID: "subscriberB"}
    
    	// 添加订阅者 A 和 B
    	object.AddSubscriber(&subscriberA)
    	object.AddSubscriber(&subscriberB)
    
    	// 更新上下文信息, 通知所有订阅者
    	object.UpdateContextMessage("Hello World!")
    
    	// 移除订阅者B
    	object.RemoveSubscriber(&subscriberB)
    
    	// 更新上下文信息,
    	object.UpdateContextMessage("Hello World Again!")
    }

    ​ 上述代码还没有一些没有去做,建议你可以尝试去实现:

    • 订阅者能够主动的订阅或者取消订阅,建议你尝试着去完成这一功能
    • 通知顺序是有序/无需的

三、模式定义

观察者模式定义了对象之间的一对多依赖,这样一来每当一个对象改变状态时,他所所有的依赖者都会收到通知并自动更新。

四、其他

​ java 中也有内置的观察者模式。java.util 包内包含最基本的 Observer 接口和 Observable 类,如果有兴趣可以去了解一下其运作方式。

相关推荐
AI+程序员在路上23 分钟前
Qt6中模态与非模态对话框区别
开发语言·c++·qt
胚芽鞘6814 小时前
关于java项目中maven的理解
java·数据库·maven
nbsaas-boot5 小时前
Java 正则表达式白皮书:语法详解、工程实践与常用表达式库
开发语言·python·mysql
岁忧5 小时前
(LeetCode 面试经典 150 题 ) 11. 盛最多水的容器 (贪心+双指针)
java·c++·算法·leetcode·面试·go
chao_7895 小时前
二分查找篇——搜索旋转排序数组【LeetCode】两次二分查找
开发语言·数据结构·python·算法·leetcode
CJi0NG5 小时前
【自用】JavaSE--算法、正则表达式、异常
java
Nejosi_念旧6 小时前
解读 Go 中的 constraints包
后端·golang·go
风无雨6 小时前
GO 启动 简单服务
开发语言·后端·golang
Hellyc6 小时前
用户查询优惠券之缓存击穿
java·redis·缓存
小明的小名叫小明6 小时前
Go从入门到精通(19)-协程(goroutine)与通道(channel)
后端·golang