Go语言中常见100问题-#13 你在项目中创建utils包了吗?

使用utils包有啥问题?

本文将讨论Go语言中一种常见的不好的实践:创建utils、common和base等共享包。首先分析这种做法存在的问题,然后讨论如何改进。

下面是一个受Go官方博客启发构造的例子,实现一个集合数据结构。在Go语言中完成该功能的惯用方法是通过 map[K]struct{} 类型来处理,K是map中允许的任何类型作为键,而值是 struct{} 类型, 表示我们对值不关心。实现代码如下,在util包中提供了两个对外的函数。

golang 复制代码
package util

func NewStringSet(...string) map[string]struct{} {

// ...

}

func SortStringSet(map[string]struct{}) []string {

// ...

}

客户端的调用方法如下:

golang 复制代码
set := util.NewStringSet("c", "a", "b")

fmt.Println(util.SortStringSet(set))

上述代码的问题是包名 util 没有任何意义,我们可以称它为 common、shared 或 base, 但它仍是一个毫无意义的名称, 无法提供任何关于包的有意义信息。我们应该创建一个富有表现力的包名称,而不是通用工具包(util),例如可以命名为 stringset.

golang 复制代码
package stringset

func New(...string) map[string]struct{} { ... }

func Sort(map[string]struct{}) []string { ... }

上面的程序去掉了 NewStringSet 和 SortStringSet 的后缀,分别变成了 New 和 Sort. 客户端调用代码变成下面的样子。

golang 复制代码
set := stringset.New("c", "a", "b")

fmt.Println(stringset.Sort(set))

NOTE:在Go语言中常见100问题-#12 项目结构最佳实践中,讨论了包的粒度问题,提到了应该避免有几十个包含一两个文件的小包。然而这种小包的思想没有问题,如果一个小的代码组具有很高的内聚性并且不真正属于其他地方,将它组织到一个特定的包中是可以接受的。也就是说包的粒度没有严格规定,找到一个平衡点即可。

解决方法

我们可以对上面的程序做进一步封装,创建一个特定的类型并将Sort作为对外提供的方法,而不是一个对外公开的函数。

golang 复制代码
package stringset

type Set map[string]struct{}

func New(...string) Set { ... }

func (s Set) Sort() []string { ... }

经过上面的重构,使得客户端调用起来更加简单,只有一个对stringset包的引用。

golang 复制代码
set := stringset.New("c", "a", "b")

fmt.Println(set.Sort())

总结

通过上面小的重构,去掉了无意义的包名,对外提供了一个富有表现力的接口。正如 Dave Cheney(Go项目组成员)所说,我们应该合理地找到处理常用程序逻辑的实用程序包。例如,如果有一个客户端和一个服务端包,应该把公共类型放在哪里呢?在这种情况下,也许一种解决方法是将客户端、服务端和公共代码组合放到一个包中。

程序包的命名是应用程序设计的一个关键点,我们应该对此保持谨慎。创建没有意义名称的共享包不是一种好的设计,像 utils、common或base包名称。此外,注意一点,以包提供的内容而不是包含的内容命名包是增加其表现力的有效方法。

相关推荐
还有你Y5 小时前
Shell 脚本语法
前端·语法·sh
踩着两条虫6 小时前
如何评价VTJ.PRO?
前端·架构·ai编程
Mh7 小时前
鼠标跟随倾斜动效
前端·css·vue.js
小码哥_常8 小时前
Kotlin类型魔法:Any、Unit、Nothing 深度探秘
前端
Web极客码9 小时前
深入了解WordPress网站访客意图
服务器·前端·wordpress
幺风10 小时前
Claude Code 源码分析 — Tool/MCP/Skill 可扩展工具系统
前端·javascript·ai编程
vjmap10 小时前
唯杰地图CAD图层加高性能特效扩展包发布
前端·gis
ZC跨境爬虫10 小时前
3D 地球卫星轨道可视化平台开发 Day7(AI异步加速+卫星系列精简+AI Agent自动评论)
前端·人工智能·3d·html·json
ID_1800790547310 小时前
淘宝 API 上货 / 商品搬家 业务场景实现 + JSON 返回示例
前端·javascript·json
M ? A10 小时前
Vue 动态组件在 React 中,VuReact 会如何实现?
前端·javascript·vue.js·经验分享·react.js·面试·vureact