

新闻资讯
技术学院Go中需显式检查指针是否为nil再解引用,因nil指针解引用会直接panic;所有可能为nil的指针(如函数返回、map查找、结构体字段等)都必须判空,常见场景包括json.Unmarshal后、HTTP请求嵌套字段、sql.NullString等。
Go 不会自动做空指针防护,nil 指针解引用直接 panic,错误信息通常是 panic: runtime error: invalid memory address or nil pointer dereference。必须在使用前显式判断。
nil,都需先判空json.Unmarshal 后未检查返回指针、接收 HTTP 请求中嵌套结构体字段、数据库查询结果为 sql.NullString 等包装类型内部指针if p != nil { use(*p) } 这类重复解引用;更安全的是提前 return 或封装为 guard clausefunc processUser(u *User) {
if u == nil {
log.Println("user is nil")
return
}
fmt.Println(u.Name) // 安全
}
声明但未初始化的指针变量默认是 nil,比如 var p *int。这类变量若被误用,极易触发 panic。应尽量让指针有明确、可控的初始化路径。
&T{} 显式取地址,而非依赖未赋值变量NewUser())应保证返回非 nil 指针,或文档注明可能返回 nil 并说明条件new(T):它只做零值分配,对复杂结构体(含嵌套指针字段)不递归初始化,仍可能产生内部 nil 字段type Config struct {
DB *sql.DB
}
// ❌ 危险:c.DB 是 nil,后续 c.DB.Query() panic
c := new(Config)
// ✅ 推荐:显式初始化关键字段,或用构造函数
c := &Config{DB: db} // db 已确认非 nil
长指针链(如 a.b.c.d.e.Name)是空引用错误的温床——任意一环为 nil 就崩。Go 更适合用组合和值语义降低间接层级。
Time 而非 *time.Time)sql.NullString、optional.Option[T](第三方库)等显式表达“存在/不存在”,而非裸指针type Order struct {
ID int
Status string
// ❌ 避免:User *User → User.Address.*string → 多层 nil 风
险
// ✅ 改为:
UserID int
Address string // 或 sql.NullString
}
单元测试常忽略指针为 nil 的分支,导致线上 panic。要主动构造 nil 输入并验证行为。
nil 输入测试用例go vet 和 staticcheck 可捕获部分明显未判空的解引用,但无法替代逻辑测试func TestProcessUser_Nil(t *testing.T) {
processUser(nil) // 应不 panic,或按约定返回 error
// 断言日志、返回值等
}
空指针不是 Go 的缺陷,而是它的显式性要求。最易被忽略的点是:**函数签名里写了 *T,不代表调用方一定会传非 nil;而你写的每个 *T 参数,都得自己负责守门**。