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包名称。此外,注意一点,以包提供的内容而不是包含的内容命名包是增加其表现力的有效方法。

相关推荐
万少7 小时前
HarmonyOS 开发必会 5 种 Builder 详解
前端·harmonyos
橙序员小站9 小时前
Agent Skill 是什么?一文讲透 Agent Skill 的设计与实现
前端·后端
炫饭第一名11 小时前
速通Canvas指北🦮——基础入门篇
前端·javascript·程序员
王晓枫12 小时前
flutter接入三方库运行报错:Error running pod install
前端·flutter
符方昊12 小时前
React 19 对比 React 16 新特性解析
前端·react.js
阿里云云原生12 小时前
5 分钟零代码改造,让 Go 应用自动获得全链路可观测能力
云原生·go
ssshooter12 小时前
又被 Safari 差异坑了:textContent 拿到的值居然没换行?
前端
曲折12 小时前
Cesium-气象要素PNG色斑图叠加
前端·cesium
Forever7_12 小时前
Electron 淘汰!新的桌面端框架 更强大、更轻量化
前端·vue.js
Angelial12 小时前
Vue3 嵌套路由 KeepAlive:动态缓存与反向配置方案
前端·vue.js