Go语言中常见100问题-#13 Creating utility packages
创建util包不是一种好的做法
本文将讨论Go语言中一种常见的不好的实践:创建utils、common和base等共享包。首先分析这种做法存在的问题,然后讨论如何改进。
下面是一个受Go官方博客启发构造的例子,实现一个集合数据结构。在Go语言中完成该功能的惯用方法是通过 map[K]struct{} 类型来处理,K是map中允许的任何类型作为键,而值是 struct{} 类型, 表示我们对值不关心。实现代码如下,在util包中提供了两个对外的函数。
package util
func NewStringSet(...string) map[string]struct{} {
// ...
}
func SortStringSet(map[string]struct{}) []string {
// ...
}
客户端的调用方法如下:
set := util.NewStringSet("c", "a", "b")
fmt.Println(util.SortStringSet(set))
上述代码的问题是包名 util 没有任何意义,我们可以称它为 common、shared 或 base, 但它仍是一个毫无意义的名称, 无法提供任何关于包的有意义信息。我们应该创建一个富有表现力的包名称,而不是通用工具包(util),例如可以命名为 stringset.
package stringset
func New(...string) map[string]struct{} { ... }
func Sort(map[string]struct{}) []string { ... }
上面的程序去掉了 NewStringSet 和 SortStringSet 的后缀,分别变成了 New 和 Sort. 客户端调用代码变成下面的样子。
set := stringset.New("c", "a", "b")
fmt.Println(stringset.Sort(set))
「NOTE:在Go语言中常见100问题-#12 Project misorganization中,讨论了包的粒度问题,提到了应该避免有几十个包含一两个文件的小包。然而这种小包的思想没有问题,如果一个小的代码组具有很高的内聚性并且不真正属于其他地方,将它组织到一个特定的包中是可以接受的。也就是说包的粒度没有严格规定,找到一个平衡点即可。」
我们可以对上面的程序做进一步封装,创建一个特定的类型并将Sort作为对外提供的方法,而不是一个对外公开的函数。
package stringset
type Set map[string]struct{}
func New(...string) Set { ... }
func (s Set) Sort() []string { ... }
经过上面的重构,使得客户端调用起来更加简单,只有一个对stringset包的引用。
set := stringset.New("c", "a", "b")
fmt.Println(set.Sort())
通过上面小的重构,去掉了无意义的包名,对外提供了一个富有表现力的接口。正如 Dave Cheney(Go项目组成员)所说,我们应该合理地找到处理常用程序逻辑的实用程序包。例如,如果有一个客户端和一个服务端包,应该把公共类型放在哪里呢?在这种情况下,也许一种解决方法是将客户端、服务端和公共代码组合放到一个包中。
程序包的命名是应用程序设计的一个关键点,我们应该对此保持谨慎。创建没有意义名称的共享包不是一种好的设计,像 utils、common或base包名称。此外,注意一点,以包提供的内容而不是包含的内容命名包是增加其表现力的有效方法。
相关文章
- 一个小巧的PHP防注入类
- PHPCMS后台入口地址隐藏方法
- 在Linux中将目录压缩打包为zip文件 并通过ftp上传的Shell脚本
- 用PHP解析XML为数组的方法
- 一段经典的抽奖算法 for PHP版
- 在Linux中用Shell脚本完成SVN版本库的建立
- PHPCMS V9无法正常通过QQ登录的解决方案
- 解决Nginx环境下PHP的flush失效问题
- 用SS5在Linux上搭建Socks5 Proxy代理服务器
- Linux中编译安装Redis和PHP扩展
- 通过iptables规则让Linux主机屏蔽Ping
- 在Linux中部署FFmpeg开源视频压缩环境过程简单笔记
- 采用单例模式编写PHP的PDO类
- Linux下管理Memcached的Service脚本
- 开源消息队列MemcacheQ在Linux中编译安装教程
- Linux下部署Memcached和PHP的Memcache扩展方法
- PHP利用FFmpeg读取视频播放时长和码率等信息
- 用PHP实现URL转换短网址的算法
- PHP通过反射方法调用执行类中的私有方法
- 在PHP中将图片转换为base64编码的方法