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
v.IsNil() == false
v.Elem() 得到目标值的反射对象(即解引用)示例:
ptr := &42
v := reflect.ValueOf(ptr)
if v.Kind() == reflect.Ptr && !v.IsNil() {
target := v.Elem() // 类型为 r
eflect.Value,代表 int(42)
fmt.Println(target.Int()) // 输出 42
}
要修改指针指向的值,v.Elem() 返回的 reflect.Value 必须是可设置的(settable),这要求原始指针本身来自一个可寻址的变量(比如局部变量、结构体字段、切片元素等)。
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 的原因:
Elem() → 先检查 !v.IsNil()
Set* → 总是检查 .CanSet()
.Kind() 和 .Type() 校验目标类型,再选对应 SetInt/SetString 等方法不复杂但容易忽略