golang 打桩,mock 数据怎么玩?
创始人
2025-06-01 13:27:39
0

工作中,很多公司都要求效能,要求自动化测试

实际落地的过程中发现,要做单元测试,自动化测试,可能当前这个服务会依赖其他服务的数据,接口等等

那么单测或者自动化的过程中,就可能会由于其他服务的原因或者环境因素导致测试失败,或者阻塞测试

这是一个问题,必须得解决,我们可以采用 golang 自带的 mock 工具来完成,可以在一些必要的地方进行数据打桩,mock 数据

gomock 是什么?

是官方提供的 一个 mock 数据的 框架

官方还提供了 mockgen 工具用来帮助 我们 生成测试代码

github 上项目地址是:https://github.com/golang/mock

官方是这样介绍 gomock 的:

gomock 是一个用于Go 编程语言的 mocking 框架。它与 Go 的内置测试包集成得很好,但也可以在其他环境中使用。

如何使用 gomock?

使用 gomock 也是非常简单的,先 go get 对应的 工具 gomock 和 mockgen

go get -u github.com/golang/mock/gomock
go get -u github.com/golang/mock/mockgen

可以写一个 demo 来进行实践

目录结构是这样的

gomock_test
├── go.mod
├── go.sum
├── main.go
└── myfunc├── mock_myfunc.go├── myfunc.go├── myuser.go└── myuser_test.go

  • mock_myfunc.go 是使用 mockgen 工具生成的
  • myfunc.go 主要是用于模拟调用的底层实现
  • myuser.go 主要是去调用 myfunc.go 里面的接口
  • myuser_test.go 是 对应的单测文件

myfunc.go

  • 编写一个 接口,里面有一个 GetInfo() string 方法,模拟获取信息
package myfunctype MyFunc interface {GetInfo() string
}

myuser.go

  • 调用 myfunc.go 中的方法,调用接口获取信息
package myfuncfunc getUser(m MyFunc) string {user := m.GetInfo()return user
}

mock 文件的生成

mock_myfunc.go

这个文件不是我们自己写的,是通过 mockgen 工具生成的 ,生成方式如下:

在 myfunc.go 的同级目录下执行如下语句,填入 source 源文件 和 目标文件即可生成新的 mock 文件

mockgen -source=myfunc.go -destination=mock_myfunc.go

我们可以看一下 mockgen 的帮助文档,还有其他的参数供我们使用

# mockgen
mockgen has two modes of operation: source and reflect.Source mode generates mock interfaces from a source file.
It is enabled by using the -source flag. Other flags that
may be useful in this mode are -imports and -aux_files.
Example:mockgen -source=foo.go [other options]Reflect mode generates mock interfaces by building a program
that uses reflection to understand interfaces. It is enabled
by passing two non-flag arguments: an import path, and a
comma-separated list of symbols.
Example:mockgen database/sql/driver Conn,Driver-aux_files string(source mode) Comma-separated pkg=path pairs of auxiliary Go source files.-build_flags string(reflect mode) Additional flags for go build.-copyright_file stringCopyright file used to add copyright header-debug_parserPrint out parser results only.-destination stringOutput file; defaults to stdout.-exec_only string(reflect mode) If set, execute this reflection program.-imports string(source mode) Comma-separated name=path pairs of explicit imports to use.-mock_names stringComma-separated interfaceName=mockName pairs of explicit mock names to use. Mock names default to 'Mock'+ interfaceName suffix.-package stringPackage of the generated code; defaults to the package of the input with a 'mock_' prefix.-prog_only(reflect mode) Only generate the reflection program; write it to stdout and exit.-self_package stringThe full package import path for the generated code. The purpose of this flag is to prevent import cycles in the generated code by trying to include its own package. This can happen if the mock's package is set to one of its inputs (usually the main one) and the output is stdio so mockgen cannot detect the final output package. Setting this flag will then tell mockgen which import to exclude.-source string(source mode) Input Go source file; enables source mode.-versionPrint version.-write_package_commentWrites package documentation comment (godoc) if true. (default true)
2021/10/30 16:43:25 Expected exactly two arguments

一般用的比较多的就是

  • -source 源文件
  • -destination 目标文件
  • -imports 依赖的需要 import 的包
  • -build_flags 传递给build工具的参数
  • -aux_files 接口文件不止一个文件时附加文件
  • -package 设置 mock 文件的包名,不设置的话,mock 文件的包名默认是 mock_输入文件的包名

通过上述指令生成的 mock 文件如下:

  • NewMockMyFunc

创建一个新的 mock 实例

  • EXPECT

允许调用者指示预期用途的对象

  • GetInfo

mock 的基础方法,也就是我们需要 mock 的方法

具体的如何使用

myuser_test.go

  • myuser.go 对应的单测文件 , 使用了 mock 的方式
package myfuncimport ("fmt""testing"gomock "github.com/golang/mock/gomock"
)func Test_getUser(t *testing.T) {mockCtl := gomock.NewController(t)mockMyFunc := NewMockMyFunc(mockCtl)mockMyFunc.EXPECT().GetInfo().Return("xiaomotong")v := getUser(mockMyFunc)if v == "xiaomotong" {fmt.Println("get user right!")} else {t.Error("get error user")}
}

看到上述单测文件,可以还不是特别明白区别,我们来看看不用 mock 的时候,我们会是如何去写单测呢

package myfuncimport ("fmt""testing"gomock "github.com/golang/mock/gomock"
)func Test_getUser(t *testing.T) {m := myfunc.CreateMyFunc() // 也就是说需要自己创建一个对象v := getUser(m)if v == "xiaomotong" {fmt.Println("get user right!")} else {t.Error("get error user")}
}

m := myfunc.CreateMyFunc() 看到上述这一句话,是创建对应的对象,再将该对象作为参数传入到 getUser 函数中,正常情况下这样做单测没有问题

但是如果这个时候创建 MyFunc 对象由于对外部还有依赖导致还没有编码好,可是也不能阻塞我们的单元测试

这个时候使用最上面的 mock 方案就显得尤为重要,可以使用 mock 的方式,mock 一个 MyFunc 对象,并设置好返回值即可完成,如:

mockCtl := gomock.NewController(t)
mockMyFunc := NewMockMyFunc(mockCtl)
mockMyFunc.EXPECT().GetInfo().Return("xiaomotong")

执行上述代码结果如下:

> go test
get user right!
PASS
ok      mygomock/myfunc 0.427s

感兴趣的朋友可以使用起来,用的多了就会更加熟悉

使用 gomock 的好处?

  • gomock 实现了较为完整的基于 interface 的 Mock 功能,能够与 Golang 内置的 testing包良好集成,也能用于其它的测试环境中
  • 学习成本低,很快就能上手

工具需要用起来,才能发挥他的价值,需要的可以用起来吧

欢迎点赞,关注,收藏

朋友们,你的支持和鼓励,是我坚持分享,提高质量的动力

好了,本次就到这里

常见技术是开放的,我们的心态,更应是开放的。拥抱变化,向阳而生,努力向前行。

我是阿兵云原生,欢迎点赞关注收藏,下次见~

相关内容

热门资讯

002201,垂直涨停!这一概... 光刻机概念股集体爆发,多股20%涨停。玻璃纤维概念股早间大幅上扬,其中九鼎新材盘中突然放量拉升,仅约...
晚间公告|12月23日这些公告... 【品大事】 8连板胜通能源:如未来股票价格进一步上涨 公司可能申请停牌核查 胜通能源公告称,公司股票...
德邦证券:2026年A股有望继... 德邦证券发布2026年投资策略报告称,2026年A股有望继续慢涨行情,稳住指数具备充足支撑,政策层面...
长期存款产品“退潮”!六大行,... 临近年末,许多人的投资储蓄需求增加,但是部分市民发现,市场上的中长期存款产品有所减少。这是怎么回事?...
关税预期与矿山停产“共振”,伦... 在供应中断预期和关税威胁的双重推动下,视为全球经济风向标的工业金属——铜再创历史性新高。 周二(12...
金价再创新高! ▽以下是正文 国际金价延续连日来的涨势。受此带动,以人民币计价的黄金价格再创新高。12月23日,有品...
华为终端软件部总裁龚体华师大开... 近日,备受高校学子关注的鸿蒙公开课・总裁校园行落地华东师范大学,华为终端BG软件部总裁龚体以“创新引...
海富通基金范庭芳: 可重点关注... 范庭芳,11年证券从业经验(其中6年为基金投资经验),复旦大学工学硕士。2015年12月加入海富通基...
关于MiniMax上市,你可能... MiniMax已经转动港交所的门把手。 12月22日,据中国证监会网站,证监会国际合作司发布关于Mi...
中国国家队与华润饮料再合作 来源:滚动播报 (来源:北京商报) 北京商报讯(记者 张君花)12月23日,“再携手 共筑体育未来”...
存量盘活 | 2025年12月... 观点指数 各省市出台十五五规划建议,市内、离岛免税双向提速期内,多省市十五五规划建议聚焦提升消费能级...
“易主方案”披露后,中国高科连... 停牌前3天涨超20%,“易主方案”出炉并复牌后,中国高科(SH600730)却连续遭遇了两个跌停。 ...
“我是股东”走进市北高新,零距... 为持续提升上市公司透明度,助力投资者洞悉企业价值,增进上市公司与投资者的深度沟通了解,2025年12...
港股风向标|连续反弹后场内情绪... 财联社12月23日讯(编辑 冯轶)今日港股午后走弱,截至收盘,恒生指数、国企指数分别下跌0.11%及...
险企求“资”若渴 发债规模处于... 新华社北京12月23日电 《中国证券报》23日刊发文章《险企求“资”若渴 发债规模处于高位》。文章称...
Ep-PEG-anisamid... 常用名称:Ep-PEG-anisamide 包装规格: 瓶装,可按 mg 或 g 级提供 Ep-PE...
金价一夜涨了36元 记者探访黄... 封面新闻记者 杨芮雯 摄影报道 12月23日,COMEX黄金期货早间站上4500美元/盎司关口,再创...
被起诉的AI独角兽,这样回应好... AIX财经(AIXcaijing)原创 作者 | 陈丹 编辑 | 魏佳 AI与版权的战争,或许正迎来...
2863亿港元完美收官 全球I... 来源:21世纪经济报道 21世纪经济报道记者张伟泽 香港报道 2025年的最后一个月,港股IPO市...
金融滴灌破解融资难题 绵阳科技... 封面新闻记者 周洪攀 如何破解科技型企业长期面临的“融资难、贷款慢”发展瓶颈难题? 12月23日下午...