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+ 索引更清晰 - 性能无优势:多一次内存加载,且增加间接层级,GC 压力略升
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。
立即学习“go语言免费学习笔记(深入)”;
- 错误现象:
panic: runtime error: invalid memory address or nil pointer dereference - 调试方法:打印
pptr和*pptr的值,确认两者都非 nil - 安全写法:
if pptr != nil && *pptr != nil { use **pptr } - 注意:Go 的
== nil可用于所有指针类型,包括**T
实际项目中,**T 出现场景极少,一旦出现,建议先问自己:能不能用接口、结构体字段或函数返回新指针替代?过度嵌套指针会让逻辑难以跟踪,尤其在并发或复杂生命周期管理中。










