Go语言中变量只有类型确定的零值而非可配置默认值,数值为0、bool为false、string为空、引用类型为nil,需区分nil与空切片、用指针或标志位判断是否显式设置。

Go 语言中没有“默认值”这个概念,只有零值(zero value)——它是类型系统强制赋予每个未初始化变量的、确定且安全的初始状态,不是可配置的默认行为,也不能被覆盖或修改。
零值不是“没值”,而是类型明确的初始值
新手常误以为 var x int 是“未定义”,其实它等价于 x := 0;var s string 就是 s := "";var b bool 就是 b := false。这不是约定,是语言规范保证的内存安全机制。
- 数值类型(
int、float64、byte等)→0或0.0 -
bool→false -
string→"" - 指针 / 切片 / 映射 / 通道 / 接口 / 函数 →
nil - 数组(如
[3]int)→ 所有元素递归设为零值,即[0, 0, 0] - 结构体 → 每个字段按其类型取零值,例如
struct{ Name string; Age int }{}等价于{"", 0}
nil 切片 vs 空切片:一个看似微小、实则致命的区别
这是最常踩坑的地方:var s []int 和 s := []int{} 都满足 len(s) == 0,但前者 s == nil 为真,后者为假。
-
var s []int:零值是nil,append(s, 1)虽然能工作(Go 自动分配),但s[0] = 1会 panic -
s := []int{}:非nil的空切片,可安全append、range、甚至cap(s) -
var m map[string]int:零值是nil,m["k"] = 1直接 panic;必须用m := make(map[string]int)或m := map[string]int{}
如何判断“用户真的设置了这个值”?零值不等于“未设置”
因为零值天然存在,你无法靠 cfg.Timeout == 0 判断用户是否显式传了 0,还是压根没填。业务上需要区分时,得主动设计:
立即学习“go语言免费学习笔记(深入)”;
- 用指针字段:
Timeout *int,nil表示未设置,*cfg.Timeout才是用户值 - 加标志位:
HasTimeout bool,配合构造函数或 setter 控制赋值路径 - 结构体标签 + 反射(谨慎):
Timeout int `default:"30"`,运行时解析 tag 并仅在值为零值时填充
new 和 make 都返回零值,但用途和返回值类型完全不同
两者都涉及零值,但语义和适用范围严格分离,混用会导致逻辑错误或编译失败:
-
new(T):适用于任意类型T,返回*T,内容为T的零值(如new([]int)返回*[]int,其值是nil切片指针) -
make(T, ...):只支持[]T、map[K]V、chan T,返回的是T本身(非指针),且完成底层数据结构初始化(如make([]int, 3)返回长度为 3、元素全为0的切片) - 常见错误:
make(int, 10)编译报错;new(map[string]int)返回*map,仍不能写入
零值机制省去了大量冗余初始化,但也意味着你必须清楚每个类型的零值表现——尤其是 nil 引用类型在读/写/收发时的不同行为边界。它不提供“智能默认”,只提供“确定初始态”。真正难的,从来不是记住 0 和 "",而是判断:这个 0,到底是用户要的,还是系统给的。










