Go中pptr(指针的指针)是**T类型,即指向*T变量的地址,该*T又指向T值,需分步声明初始化,仅在需修改指针变量自身(如置nil)时使用,须防nil解引用。
**pptr(指针的指针)?Go 中没有“多级指针”的语法糖,但可以通过嵌套声明实现指针的指针,即 **T 类型。它本质是一个指向 *T 变量的地址,而该 *T 又指向一个 T 值。这不是 C 风格的任意级间接访问,而是明确、类型安全的两级解引用。
**int?必须分步完成:先有值,再取地址得 *int,再对该指针变量取地址得 **int。不能直接对字面量或临时表达式取地址。
&i 合法,i 是变量;&(&i) 合法,因为 &i 是可寻址的变量(存储在栈上)&(&42) 非法 —— 字面量 42 不可寻址,无法对其地址再取地址var p **int; *p = &x 会 panic:nil 指针解引用func main() {
x := 100
ptr := &x // *int
pptr := &ptr // **int
fmt.Println(**pptr) // 输出 100
}
**T?绝大多数 Go 场景用不到。只有当函数需修改「一个指针变量本身」(而非它指向的值)时才需 **T。典型场景是重置外部指针为 nil 或指向新分配对象。
resetPtr(pptr *string) 只能改 *string 指向的字符串内容;要让调用方的 strPtr 变成 nil,必须传 **string
**T 实现“动态修改多个不同指针”通常说明设计可重构,比如改用切片 []*T + 索引更清晰func setToNil(s **string) {
*s = nil // 修改调用方传入的指针变量本身
}
func main() {
name := "hello"
ptr := &name
setToNil(&ptr)
fmt.Println(ptr == nil) // true
}
运行时 panic 往往源于未检查 **T 是否为 nil。Go 不做空指针防护,**pptr 解引
用前必须确保 pptr != nil 且 *pptr != nil。
panic: runtime error: invalid memory address or nil pointer dereference
pptr 和 *pptr 的值,确认两者都非 nilif pptr != nil && *pptr != nil { use **pptr }
== nil 可用于所有指针类型,包括 **T
实际项目中,**T 出现场景极少,一旦出现,建议先问自己:能不能用接口、结构体字段或函数返回新指针替代?过度嵌套指针会让逻辑难以跟踪,尤其在并发或复杂生命周期管理中。