要通过 Go 的 reflect 包获取指针指向的底层值,必须先用 reflect.ValueOf(&x) 获取可寻址的指针反射值,再调用 Elem() 解引用;嵌套指针需循环判断 Kind() == reflect.Ptr 且非 nil 后多次 Elem();读写需确保来源可寻址,安全做法是封装 Deref 函数自动穿透非 nil 指针。

要通过 Go 的 reflect 包获取指针指向的底层值,核心是理解反射中“可寻址性”与“解引用”的关系——不能直接对非可寻址的 reflect.Value 调用 Elem(),必须先确保其底层值可被修改(即来自变量、切片元素、结构体字段等),再安全解引用。
1. 基础解引用:从指针变量开始
只有当 reflect.Value 表示一个**可寻址的指针**(比如取自变量地址)时,才能调用 Elem() 获取所指对象的值:
- ✅ 正确做法:用
reflect.ValueOf(&x)得到指针的反射值,再调用.Elem()拿到x的值 - ❌ 错误做法:对
reflect.ValueOf(x)(x 是指针类型)直接调用.Elem()—— 这会 panic,因为该 Value 不可寻址
示例:
var x = 42ptr := &x
v := reflect.ValueOf(ptr) // 可寻址的 *int
val := v.Elem() // int 类型的 Value,值为 42
2. 处理嵌套指针或多重间接
如果指针指向另一个指针(如 **int),需多次调用 Elem(),但每次都要检查是否仍为指针类型且可解引用:
立即学习“go语言免费学习笔记(深入)”;
- 用
v.Kind() == reflect.Ptr判断是否为指针 - 用
v.IsNil()避免对 nil 指针调用Elem()(否则 panic) - 循环调用
v = v.Elem()直到不再是Ptr类型
常见用途:通用深拷贝、日志打印、ORM 字段解析等需要穿透多层指针的场景。
3. 解引用后如何读写值
v.Elem() 返回的是被指向值的反射表示,若需读写,要注意:
- 读取:直接用
v.Elem().Int()、.String()等方法(前提是类型匹配) - 写入:必须保证原始指针来自可寻址变量(如
&x),且目标类型允许赋值;调用v.Elem().Set(xxx)才会真正修改原变量 - 不可写情况举例:对字面量指针(如
reflect.ValueOf(new(int)).Elem())可以写,但对reflect.ValueOf(&42).Elem()会失败——因为&42是不可寻址的临时地址(Go 不允许取字面量地址)
4. 安全解引用封装建议
生产代码中推荐封装一个健壮的解引用函数,避免 panic:
func Deref(v reflect.Value) reflect.Value {for v.Kind() == reflect.Ptr {
if v.IsNil() {
return reflect.Value{} // 或 panic/返回零值
}
v = v.Elem()
}
return v
}
这个函数能自动穿透任意深度的非 nil 指针,返回最底层的非指针值,适用于泛型工具、调试器、序列化库等。
基本上就这些。Golang 反射的指针解引用不复杂但容易忽略可寻址性前提,抓住 ValueOf(&x) 和 Elem() 的配合逻辑,就能稳稳拿到底层值。










