注:本文已发布超过一年,请注意您所使用工具的相关版本是否适用
本系列为 Go 进阶训练营 笔记,访问 博客: Go进阶训练营, 即可查看当前更新进度,部分文章篇幅较长,使用 PC 大屏浏览体验更佳。
先简单回顾一下 《Go 工程化(九) 项目重构实践》 如果还没看过之前这篇文章可以先看一下

在我们之前的项目目录分层中,我们主要分为了五个块
cmd/appname 是我们服务的入口,只负责启动和依赖注入(使用 Wire)domain 或者 model 是我们的实体定义 + 接口定义server 负责实现我们在 proto 中定义的接口,在这一层中我们只做数据转换,不写业务逻辑usecase 负责实现我们的业务逻辑repo 负责数据操作, 仅做数据操作,不实现业务逻辑在之前的文章中仅仅提到了一个非常简单的示例,但是我们实际业务流程往往没有那么简单,就一个非常常见的例子,我们现在需要创建一篇文章,文章上需要关联分类或者是标签信息,这里至少就分两步
这两个创建操作需要保证一致性,我们需要在数据库中使用事务,这时候我们的事务在哪里承载呢?
其中最简单也最直接的办法就是在 repo 的 CreateArticle 方法中我们就使用事务去同时创建文章以及标签之间的关联关系。
针对第一个问题,最简单的办法就是我们提供一个 CreateArticleWithTags 方法表示同时创建这两者,如果我们需要一个独立的 CreateArticle 再写一个就好了。
但是随着需求越来越多,可能后面还有需要和角色关联的,和商品关联的等等
ok,所以直接在 repo 层里面来实现看上去好像行不通,那我们就把视线往上移动,我们在 usecase 来解决这个问题
事务的能力是在 repo 上提供的,所以我们需要在 repo 层提供一个事务接口,然后在 usecase 中进行调用,保证是事务执行的就行
1 | |
在 repo 中,我们可以像上面这样定义,提供一个 Tx 方法,这个方法接受一个 ArticleRepoTxFunc 作为参数,这个函数中的 repo 是开启了事务的 repo,通过这个 repo 调用的所有方法都是在事务中执行的
1 | |
然后我们在 usecase 调用的时候就可以这样
1 | |
这样写起来就整洁很多了,业务逻辑和我们最初的设计一样,在 usecase 中实现了,repo 中我们也保持了简单的原则。
这样是不是就万事大吉了呢?如果万事大吉了这篇文章到这儿也就应该结束了,但是还没有,说明我在实践的过程中还碰到了问题
问题很简单,就是我们在 usecase 中不仅仅需要复用 repo 中的代码,还有可能需要复用 usecase 中的代码,不然我们就可能在 usecase 中出现很多相同的逻辑代码片段,代码的重复率就很高
我们来看下面一个例子会不会发现有点什么不对
1 | |
这个方法内其实是开启了两个事务,这两个事务之间互不相关,不符合我们需求
1 | |
我们在 usecase 中也创建了一个 tx 方法,和 repo 类似,在调用 tx 之后,handler 中的方法的需要都是用新的参数 usecase 这个新的 usecase 可以保证里面的 repo 调用都是事务的
所以我们之前的 A 函数可以修改为这样
1 | |
这样就没有问题了么?我们 UpdateXXX 方法中也调用 u.tx 方法,这样就会导致反复开启事务,虽然在 gorm 的 Transaction 方法是支持嵌套事务的,但是我们还是不要滥用这个特性。
解决办法很简单,我们只需要在执行的时候判断下就行了
1 | |
然后我们在 tx 方法内
1 | |
Tx 方法由于 repo.Tx 本身也是接口的一个方法,所以其实我们只需要对 repo.Tx 进行 mock 就行了
1 | |
细心的朋友可能会说,这个测试等于没有测试啊,代码都没有覆盖到,直接就返回结果了
是的,这种情况两种处理办法,一种是就按照上面这么写。然后我们把事务内的函数单独提出来作为一个方法,这样就都能覆盖到,并且独立测试了
1 | |
还有一种是我们在 mock repo tx 方法的时候换一种写法, 如下所示
1 | |
文章到这里就到尾声了,同样的问题,我们现在这么写就可以了么?
对于我当前所遇到的一些需求来说已经可以解决了,当然这个方案并不完美,比如说我们涉及到多个 repo 的时候,当前的方法就没法直接用了,还得进行一些改造,虽然我们要有远见但是也不要想的太多,进化是优于完美的,如果你有什么好的想法欢迎在评论区探讨
一个小预告,接下来还有几篇工程实践的相关文章
注:本文已发布超过一年,请注意您所使用工具的相关版本是否适用
本系列为 Go 进阶训练营 笔记,访问 博客: Go进阶训练营, 即可查看当前更新进度,部分文章篇幅较长,使用 PC 大屏浏览体验更佳。
先简单回顾一下 《Go 工程化(九) 项目重构实践》 如果还没看过之前这篇文章可以先看一下

在我们之前的项目目录分层中,我们主要分为了五个块
cmd/appname 是我们服务的入口,只负责启动和依赖注入(使用 Wire)domain 或者 model 是我们的实体定义 + 接口定义server 负责实现我们在 proto 中定义的接口,在这一层中我们只做数据转换,不写业务逻辑usecase 负责实现我们的业务逻辑repo 负责数据操作, 仅做数据操作,不实现业务逻辑在之前的文章中仅仅提到了一个非常简单的示例,但是我们实际业务流程往往没有那么简单,就一个非常常见的例子,我们现在需要创建一篇文章,文章上需要关联分类或者是标签信息,这里至少就分两步
这两个创建操作需要保证一致性,我们需要在数据库中使用事务,这时候我们的事务在哪里承载呢?
其中最简单也最直接的办法就是在 repo 的 CreateArticle 方法中我们就使用事务去同时创建文章以及标签之间的关联关系。
针对第一个问题,最简单的办法就是我们提供一个 CreateArticleWithTags 方法表示同时创建这两者,如果我们需要一个独立的 CreateArticle 再写一个就好了。
但是随着需求越来越多,可能后面还有需要和角色关联的,和商品关联的等等
ok,所以直接在 repo 层里面来实现看上去好像行不通,那我们就把视线往上移动,我们在 usecase 来解决这个问题
事务的能力是在 repo 上提供的,所以我们需要在 repo 层提供一个事务接口,然后在 usecase 中进行调用,保证是事务执行的就行
1 | |
在 repo 中,我们可以像上面这样定义,提供一个 Tx 方法,这个方法接受一个 ArticleRepoTxFunc 作为参数,这个函数中的 repo 是开启了事务的 repo,通过这个 repo 调用的所有方法都是在事务中执行的
1 | |
然后我们在 usecase 调用的时候就可以这样
1 | |
这样写起来就整洁很多了,业务逻辑和我们最初的设计一样,在 usecase 中实现了,repo 中我们也保持了简单的原则。
这样是不是就万事大吉了呢?如果万事大吉了这篇文章到这儿也就应该结束了,但是还没有,说明我在实践的过程中还碰到了问题
问题很简单,就是我们在 usecase 中不仅仅需要复用 repo 中的代码,还有可能需要复用 usecase 中的代码,不然我们就可能在 usecase 中出现很多相同的逻辑代码片段,代码的重复率就很高
我们来看下面一个例子会不会发现有点什么不对
1 | |
这个方法内其实是开启了两个事务,这两个事务之间互不相关,不符合我们需求
1 | |
我们在 usecase 中也创建了一个 tx 方法,和 repo 类似,在调用 tx 之后,handler 中的方法的需要都是用新的参数 usecase 这个新的 usecase 可以保证里面的 repo 调用都是事务的
所以我们之前的 A 函数可以修改为这样
1 | |
这样就没有问题了么?我们 UpdateXXX 方法中也调用 u.tx 方法,这样就会导致反复开启事务,虽然在 gorm 的 Transaction 方法是支持嵌套事务的,但是我们还是不要滥用这个特性。
解决办法很简单,我们只需要在执行的时候判断下就行了
1 | |
然后我们在 tx 方法内
1 | |
Tx 方法由于 repo.Tx 本身也是接口的一个方法,所以其实我们只需要对 repo.Tx 进行 mock 就行了
1 | |
细心的朋友可能会说,这个测试等于没有测试啊,代码都没有覆盖到,直接就返回结果了
是的,这种情况两种处理办法,一种是就按照上面这么写。然后我们把事务内的函数单独提出来作为一个方法,这样就都能覆盖到,并且独立测试了
1 | |
还有一种是我们在 mock repo tx 方法的时候换一种写法, 如下所示
1 | |
文章到这里就到尾声了,同样的问题,我们现在这么写就可以了么?
对于我当前所遇到的一些需求来说已经可以解决了,当然这个方案并不完美,比如说我们涉及到多个 repo 的时候,当前的方法就没法直接用了,还得进行一些改造,虽然我们要有远见但是也不要想的太多,进化是优于完美的,如果你有什么好的想法欢迎在评论区探讨
一个小预告,接下来还有几篇工程实践的相关文章