如何在Golang中通过反射获取指针底层值_读取和修改指针指向

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),这要求原始指针本身来自一个可寻址的变量(比如局部变量、结构体字段、切片元素等)。

  • 不可行: 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 等方法

不复杂但容易忽略