iota 是 Go 编译期行号计数器,用于安全定义枚举:从 0 开始用 const 块逐行声明;跳过 0 用 _ = iota 占位;权限标志需结合位运算(如 1
iota 是 Go 中唯一能自动递增的常
量生成器,它不是语法糖,而是编译期确定的行号计数器——用对了省心,写错一行就全偏移。
最常见也最容易上手的用法:在 const 块中逐行声明,iota 自动从 0 起步、每行 +1。
const ( Running = iota // 0 Paused // 1 Stopped // 2 Restarting // 3 )
Paused),就隐式复用上一行的完整表达式(即 Running 的值 + 当前行 iota 增量)Running State = iota)也能工作,但类型安全弱,反序列化或传参时易出错Paused 后加 Resuming),所有后续值都会变——这是反序列化失败的高发原因Go 中整型零值(0)常被解释为“未初始化”或“无效状态”,比如 Status(0) 可能被误判为 Unknown 而非 Active。所以工程中普遍避开 0。
const ( _ = iota // 丢弃 0,不导出、不参与逻辑 Active // 1 Inactive // 2 Pending // 3 )
_ = iota 占位是最清晰的跳过方式,比 Active = iota + 1 更易读且不易漏掉后续项Active = 1; Inactive = 2 手动赋值——失去 iota 的自维护性,删/增项时极易遗漏更新_ = iota + 99,但需加注释说明意图当枚举需要支持“多选组合”(如用户权限 Read | Write)或“按位判断”(perm & Read != 0),必须用左移生成 2 的幂。
type Permission int const ( Read Permission = 1 << iota // 1 Write // 2 Delete // 4 Admin // 8 )
Read | Delete 得到 5(二进制 101),可无损组合Read = iota; Write = iota + 1 —— 这样得到的是连续整数,无法做位运算判断a, b = iota, iota+1),该行 iota 值相同,需注意是否符合预期裸 int 枚举在日志、调试、API 返回时全是数字,可读性差,也不利于 IDE 自动补全和类型检查。
type Status int
const (
Unknown Status = iota // 0
OK // 1
Error // 2
Timeout // 3
)
func (s Status) String() string {
switch s {
case Unknown:
return "unknown"
case OK:
return "ok"
case Error:
return "error"
case Timeout:
return "timeout"
default:
return "status?"
}
}
fmt.Printf("%v", OK) 会输出 ok,而不是 1
Status 能阻止把任意 int 赋给它,比如 var s Status = 999 编译不通过(除非强制转换)default 分支——万一将来新增枚举值但忘了加 String() case,至少不会 panic真正容易被忽略的点是:iota 不跨 const 块重用,但人会跨块复制粘贴。复制一个枚举块到新文件时,若忘记删掉旧的 _ = iota 占位,新块就会整体偏移——这种 bug 往往只在特定环境触发,排查成本远高于写时多看一眼。