Go中变量声明即初始化,零值是类型确定的默认值:数值类型为0,bool为false,string为空字符串,指针等为nil,struct各字段取零值,数组元素全为零;time.Time零值为0001-01-01 UTC,需用IsZero()判断;reflect.ValueOf(nil)会panic,应传带类型的nil或直接比较零值。

Go 中变量声明即初始化,所谓“零值”就是它被自动赋予的、类型确定的默认值——不是随机值,也不是未定义行为,而是语言强制保证的安全起点。
哪些值算零值?看类型直接对号入座
零值不是统一的 nil 或 0,而是按类型严格定义的:
-
int、float64、uint8等数值类型 →0(或0.0、0+0i) -
bool→false -
string→""(空字符串,不是nil) -
*T、func()、interface{}、map[K]V、[]T、chan T→nil -
struct{}→ 所有字段各自取零值(如type S struct{ N string; A int }的零值是{N: "", A: 0}) -
[3]int→[0, 0, 0](数组是值类型,每个元素都初始化)
注意:time.Time 是结构体,它的零值是 0001-01-01 00:00:00 +0000 UTC,不能用 == nil 判断,得用 t.IsZero()。
为什么不能直接用 reflect.ValueOf(nil) 检查零值?
因为 nil 接口没有类型信息,reflect.ValueOf(nil) 会 panic。这是反射最常踩的坑之一。
立即学习“go语言免费学习笔记(深入)”;
正确做法是确保传入带类型的 nil 值:
reflect.ValueOf((*int)(nil)).IsZero() // ✅ true
reflect.ValueOf((map[string]int)(nil)).IsZero() // ✅ true
reflect.ValueOf(struct{}{}).IsZero() // ✅ true
reflect.ValueOf(nil) // ❌ panic: invalid reflect.Value
更推荐的方式:对已知类型,直接比较零值(如 s == ""、v == 0),既快又清晰;只有泛型或配置校验等少数场景才用反射。
零值不是“没设置”,而是“已设置为默认”
很多新手误以为 var m map[string]int 是“空 map”,结果一写就 panic:
var m map[string]int m["key"] = "value" // panic: assignment to entry in nil map
这是因为 m 的零值是 nil,不是空 map。必须显式 make 才能使用:
-
map、slice、chan的零值是nil,不可读写(除len/cap对 slice 安全) -
sync.Mutex、sync.WaitGroup等标准库类型依赖零值可用——声明即能用,无需new或make - 结构体字段若未赋值,自动为零值,但业务上可能需要区分“用户没填”和“填了空字符串/0”,这时得加额外标记字段或用指针包装(如
*string)
零值机制省去了大量手动初始化代码,但它不解决语义问题:一个 0 是用户明确设的,还是系统给的默认?这点在 API 请求解析、配置合并、数据库字段映射时尤其关键,不能只看值,还得结合上下文判断。










