0

0

如何使用Golang实现状态模式_Golang状态模式State Pattern行为切换

P粉602998670

P粉602998670

发布时间:2026-01-17 10:23:02

|

695人浏览过

|

来源于php中文网

原创

Go中State接口设计核心是用接口定义行为契约、结构体字段存状态、指针接收者动态替换状态;必须避免值传递、空指针、竞态,统一通过SetState切换并校验nil。

如何使用golang实现状态模式_golang状态模式state pattern行为切换

什么是 Go 里的 State 接口设计核心

Go 没有类和继承,所以状态模式不能照搬 Java/C# 的抽象类 + 子类继承写法。关键在于:用接口定义行为契约,用结构体字段保存当前状态,并通过指针接收者方法动态替换自身状态。

典型错误是把状态当作值类型传参或复制,导致状态切换不生效;正确做法是让上下文(如 *VendingMachine)持有指向状态接口的指针,并在状态变更时更新该指针。

  • State 必须是接口,至少包含当前业务需要响应的行为方法(如 InsertCoin()PressButton()
  • 每个具体状态实现为独立结构体,方法接收者必须是指针(func (s *HasCoinState) InsertCoin(m *VendingMachine)),否则无法修改 m.state
  • 上下文结构体里状态字段类型是 State 接口,不是具体类型

如何安全地在状态间切换而不引发 panic

常见 crash 场景是状态方法里调用了尚未初始化的上下文字段,或在切换过程中出现竞态(尤其并发调用时)。Go 中最稳妥的做法是:所有状态变更只通过上下文提供的统一入口(如 SetState(s State)),并在其中做 nil 检查和原子赋值。

例如,避免直接写 m.state = &SoldOutState{},而应封装为:

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

func (m *VendingMachine) SetState(s State) {
    if s == nil {
        panic("state cannot be nil")
    }
    m.state = s
}
  • 每次状态变更前检查 s != nil,防止空指针解引用
  • 如果涉及并发访问m.state 字段需用 sync/atomic.Value 或互斥锁保护
  • 不要在状态方法内部直接 new 新状态并赋值给 m.state,应由上下文统一调度,便于测试和拦截

switch 和状态接口哪个更适合行为分支

switch 枚举状态类型(如 type StateType int; const HasCoin StateType = iota)看似简单,但会破坏状态模式的核心价值:开闭原则。一旦新增状态,所有 switch 处都要改,且无法封装各自逻辑。

NewsBang
NewsBang

盛大旗下AI团队推出的智能新闻阅读App

下载

接口方式虽然多写几个文件,但换来的是可插拔性——比如测试时可注入 mock 状态,运维时可动态加载新状态实现。

  • switch 仅适用于状态极少(≤3)、逻辑极简、且确定永不扩展的场景
  • 接口方式下,每个状态的副作用(如日志、通知、DB 更新)完全隔离,不会污染其他状态分支
  • 注意:Go 的接口是隐式实现,无需声明 implements,但结构体字段命名要一致(如都含 machine *VendingMachine 才能复用公共逻辑)

为什么常在状态方法里传入 *VendingMachine 而非只传数据

因为状态行为往往需要触发上下文的副作用:扣减库存、发消息、重置计时器、切换到下一个状态。如果只传原始数据(如 coinCount int),状态实现就变成纯函数,无法驱动系统流转。

典型例子:SoldState.PressButton() 需要调用 m.ReleaseItem() 并立即设为 SoldOutState,这两步必须发生在同一上下文实例上。

  • *VendingMachine 是为了获得「执行权」,不是为了读取字段——尽量减少状态对上下文内部字段的直接访问
  • 若担心循环引用,可将上下文需暴露的能力抽成小接口(如 ReleaserResetter),状态只依赖接口而非具体结构体
  • 切勿在状态方法中启动 goroutine 并异步修改 m.state,这极易导致状态错乱,应改为同步调用 m.SetState()

状态模式在 Go 里真正难的不是语法,而是克制——忍住不用 switch,忍住不在状态里直接操作上下文私有字段,忍住不把状态逻辑塞进一个大结构体里。每多一层间接,就多一分可维护性。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

834

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

739

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

735

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.9万人学习

Java 教程
Java 教程

共578课时 | 46.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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