Go中指针零值为nil,解引用前必须判空,否则panic;接口nil判断需注意类型与值双空;结构体指针字段需分层判空,跨边界场景尤需警惕。
Go 中指针的零值就是 nil,不是“未定义”,也不是“空地址”,而是语言明确定义的、可安全比较的零值。解引用前不判断 ptr == nil,程序必然 panic。
Go 的指针是显式类型,*T 类型变量未初始化或显式赋为 nil 时,其值就是 nil。对它执行 *ptr 操作会触发运行时 panic:invalid memory address or nil pointer dereference。
json.Unmarshal 返回 *T)可能为 nil,尤其在解析失败或字段缺失时type User struct{ Profile *Profile }),即使 user 非 nil,user.Profile 仍可能为 nil
用 == nil 是最直接、最推荐的方式。Go 规范保证所有指针类型都支持与 nil 比较,且该操作常量时间、无副作用。
if ptr != nil { name := *ptr }
if ptr != (*T)(nil) { ... } —— 不必要,且易出错nil 不能和非指针类型比较,比如 int、string、time.Time 都不能和 nil 比(编译报错)nil,但它的 nil 判断逻辑完全不同——不要把 io.Writer 接口变量当成普通指针来判空这是 Go 中最典型的 nil 陷阱。接口变量由两部分组成:动态类型(type)和动态
值(value)。只有两者同时为空,接口才整体为 nil。
var w io.Writer = (*bytes.Buffer)(nil); if w == nil { ... } → 实际上 w != nil,因为 type 是 *bytes.Buffer
if b, ok := w.(*bytes.Buffer); ok && b == nil { ... }
error(接口类型),你不能靠 err == nil 就认为底层没发生错误;但如果函数内部返回了 fmt.Errorf("..."),那 err 就是非 nil 接口,没问题;真正危险的是返回了 (*MyError)(nil) 这类构造结构体本身是值类型,零值是各字段零值的组合;其中指针字段的零值就是 nil,但它不影响结构体变量本身的可访问性。
if user != nil && user.Profile != nil { name := user.Profile.Name }
nil ≠ 字段非 nil;user 是局部变量?它根本不可能是 nil(除非是 *User)reflect + IsNil(),但更推荐业务层显式校验,避免反射开销和可读性损失真正难的不是“怎么写 if ptr == nil”,而是意识到哪些地方 *可能* 出现 nil——尤其是跨包调用、JSON 解析、数据库扫描、HTTP 响应反序列化这些边界场景。一旦漏掉一层判空,panic 就在下一行等着你。