nil 是 Go 中引用类型的零值,解引用前必须判 nil,否则必 panic;需在函数入口、方法体内、字段访问前手动检查,接口 nil 判定需类型和值均为零值,泛型 Deref 可安全读取但不解决设计问题。
nil 在 Go 中不是“空值通用占位符”,而是指针、切片、map 等引用类型的**零值**;对 *T 类型指针解引用前不判 nil,必 panic——这是最常踩的坑,也是唯一必须守住的底线。
Go 不会自动跳过 nil 指针访问,*p 一旦 p == nil 就崩溃,没有“安全导航”语法(如 JavaScript 的 ?. )。必须手动防护:
if p == nil 检查,尤其接收外部传入的指针参数时if u == nil,否则 var u *User; u.Greet() 直接挂掉Name *string)时,即使结构体非 nil,该字段也可能为 nil,需单独判断func printName(u *User) {
if u == nil {
fmt.Println("user is nil")
return
}
if u.Name != nil {
fmt.Println(*u.Name)
} else {
fmt.Println("name not set")
}
}
interface{} 和 error 的 nil 很容易误判接口变量为 nil 的条件是:**动态类型 + 动态值都为零值**。只要类型存在,哪怕底层值是 nil,接口本身就不等于 nil:
var err *MyError = nil; return err → 返回的是 type=*MyError, value=nil,接口不为 nil,调用方 if err != nil 会误判为有错误var w io.Writer = (*bytes.Buffer)(nil) → w == nil 是 false,但 w.Write(...) 会 panic正确做法:返回 return nil(接口零值),而不是返回一个 nil 指针赋给接口。
频繁写 if p != nil { *p } 易漏、冗余。可用泛型封装兜底逻辑
:
func Deref[T any](ptr *T, def T) T {
if ptr == nil {
return def
}
return *ptr
}
name := Deref(namePtr, "anonymous") // 安全取值,无需每次判空
nil 指针问题往往始于源头:
var p *int 默认就是 nil,但很多人误以为“没写 = nil 就不是 nil”var u *User; return u),调用方拿到的就是 nil
真正关键的不是“怎么修 panic”,而是“在哪初始化、谁负责判空、谁该返回 nil”。这些责任边界一旦模糊,nil 就成了定时炸弹。