go - 您是否应该使用零“枚举”值来指示无效值
问题描述
使用 C 几十年后,我养成了使用枚举的零值作为特殊的未定义/未知/错误值的习惯。多年来,我相信这为我节省的调试时间不是数小时甚至数天,而是数月,因为它使值未初始化时很明显。(对于存在合理默认值且不可能存在未初始化值的简单枚举,我不会这样做。)
在我看来,这种做法在 Go 中更有用,因为值会自动为您初始化为零。但是,有人告诉我“惯用的”Go 零值应该是有效值。我认为这个“规则”是为结构发明的,在没有构造函数的情况下,有一个新创建的“归零”结构可供使用是很有意义的,但是在某些情况下没有逻辑默认值(对于结构和枚举)。
如果你需要它,这里是一个例子:
type Base int
const (
Invalid Base = iota
A
C
T
G
)
请注意,我已经在 SO 上广泛搜索了这个问题,并对这个特定主题没有被涵盖感到惊讶。我意识到我的问题有些主观,可能会被标记,但我认为它很有用。我正在寻找证据表明使用零值来指示错误条件是可接受的 Go 实践。这种用法的任何例子,例如。来自标准 Go 库,将不胜感激。
解决方案
真正的enum
类型应该只从预定义的常量值列表中分配一个值。然而,该go
语言没有这样的类型值强制。
go
hasconst
它通常使用 say 的派生类型int
。没有编译/运行时机制来强制一个值严格在预定义的列表内。
那么这在实践中意味着什么?
您的枚举值是强制性的还是可选的?也就是说,在反序列化“枚举”值时,是不是:
- 可选 - 然后使用零值表示默认值
- 强制 - 然后零值表示初始化错误
根据您的常见用例,选择这两个选项之一。
编辑: 反序列化不是唯一的问题。在枚举值上进行分支时必须小心。例如:
type role int
const (
user role = iota
helpdesk
admin
)
func greet(r role) {
switch r {
case admin:
fmt.Println("hi admin")
case helpdesk:
fmt.Println("hi helpdesk")
default:
fmt.Println("hi user") // right?
}
}
这有效:
var r role
r = admin
greet(r) // hi admin
但是这个呢?
r = 12
greet(r) // 'hi user' ?!!
因此,请务必仅对有效值进行迂腐验证:
func validateRole(r role) (err error) {
switch r {
case user, helpdesk, admin: // all valid values
default:
err = fmt.Errorf("invalid `role` enum %d", r)
}
return
}
推荐阅读
- python - wrap_text 与 openpyxl。如何使用文档来解决弃用警告?
- c - 从 GPIO 输入驱动程序公开 /dev/uio* 文件
- c# - 无法导航到插入符号下的符号 - Visual Studio 2015、2017 和 2019
- sql - 如何搜索数据以查找没有特定值的记录?
- php - 我正面临这个错误“致命错误:未捕获的错误:找不到类'Zend\Uri\Uri'”
- spring-boot - 是否可以在springboot中从命令行设置参数spring.servlet.multipart.max-file-size?
- python - 如何在python中返回一个json块
- python-3.x - 为什么这些 LSTM 参数计数不同?
- r - 我将如何反转 R 中矩阵的二进制数字(0 和 1)?
- c++ - boost odeint 中的受控错误步进器是否支持复杂的数据类型?