Go反射操作指针需先检查Kind为Ptr且非nil,再通过Elem()读取值;修改时还需确保Elem()结果CanSet(),否则panic。

在 Go 中,反射(reflect 包)可以用于动态读取和修改指针所指向的值,但必须注意:只有可寻址(addressable)的指针才能被修改。直接对非可寻址的指针(如字面量取地址、函数返回的临时指针)调用 Set* 方法会 panic。
获取指针底层值(读取)
使用 reflect.Value.Elem() 获取指针指向的值的 reflect.Value。前提是该值本身是有效的指针类型且非 nil。
- 先用
reflect.ValueOf(x)得到原始值的反射对象 - 检查是否为指针:
v.Kind() == reflect.Ptr - 检查是否非 nil:
v.IsNil() == false - 调用
v.Elem()得到目标值的反射对象(即解引用)
示例:
ptr := &42
v := reflect.ValueOf(ptr)
if v.Kind() == reflect.Ptr && !v.IsNil() {
target := v.Elem() // 类型为 reflect.Value,代表 int(42)
fmt.Println(target.Int()) // 输出 42
}
修改指针指向的值(写入)
要修改指针指向的值,v.Elem() 返回的 reflect.Value 必须是可设置的(settable),这要求原始指针本身来自一个可寻址的变量(比如局部变量、结构体字段、切片元素等)。
立即学习“go语言免费学习笔记(深入)”;
- 不可行:
reflect.ValueOf(&42).Elem().SetInt(100)→ panic(不可设置) - 可行:
var x = 42; reflect.ValueOf(&x).Elem().SetInt(100)→ 成功,x 变为 100
安全写法:
var num = 10
ptr := &num
v := reflect.ValueOf(ptr)
if v.Kind() == reflect.Ptr && !v.IsNil() {
elem := v.Elem()
if elem.CanSet() {
elem.SetInt(99)
}
}
fmt.Println(num) // 输出 99
处理嵌套指针与接口中的指针
若指针藏在接口或结构体中,需逐层解包:
- 接口值传入
reflect.ValueOf(interface{})后,需先.Elem()(如果接口持有一个指针) - 结构体字段为指针时,先用
.Field(i)获取字段值,再判断是否为指针并.Elem()
例如:
type Person struct {
Age *int
}
age := 25
p := Person{Age: &age}
v := reflect.ValueOf(p).FieldByName("Age")
if v.Kind() == reflect.Ptr && !v.IsNil() && v.Elem().CanSet() {
v.Elem().SetInt(30)
}
// age 现在是 30
常见错误与规避方式
反射操作指针最常遇到 panic 的原因:
-
对 nil 指针调用
Elem()→ 先检查!v.IsNil() -
对不可设置的值调用
Set*→ 总是检查.CanSet() -
类型不匹配 → 使用
.Kind()和.Type()校验目标类型,再选对应SetInt/SetString等方法
不复杂但容易忽略










