指针类型的反射操作需要注意类型安全和可寻址性。当你拿到一个reflect.value表示指针类型时,必须使用elem()方法解引用才能访问实际值,且只能对ptr类型调用elem(),否则会panic;如果你有一个非指针类型的reflect.value但需要获取其指针,需使用addr()方法,前提是该值必须是addressable,否则也会panic。1.elem()用于从指针获取值,2.addr()用于从值获取指针,两者常配合使用,如通过reflect.valueof(&s).elem()获得可修改的结构体value以便操作字段。使用时必须先判断kind是否为ptr,并确认值是否可寻址,以避免运行时错误。

在 Go 语言的反射机制中,指针类型是一个常见的处理对象。由于 Go 的反射设计强调类型安全性,所以在操作指针时会有一些特殊处理方式,尤其是 Elem() 和 Addr() 这两个方法经常被用到。理解它们的作用和使用场景,能帮助我们更准确地进行反射操作。

指针类型的反射操作需要注意什么?
当你拿到一个 reflect.Value 表示的是指针类型时,不能直接访问它指向的值或修改其内容。这时候就需要用到 Elem() 方法来获取指针指向的实际值。如果原始值不是一个指针,调用 Elem() 会导致 panic,所以通常需要先检查是否是指针类型。

反过来,如果你有一个非指针类型的 reflect.Value,但你想通过反射来设置它的值(比如在结构体字段赋值时),你就需要用到 Addr() 方法,来获得一个指向该值的指针。
立即学习“go语言免费学习笔记(深入)”;
如何正确使用 Elem() 获取指针指向的值?
当一个变量是通过指针传入反射系统时,你需要使用 Elem() 来“解引用”才能访问实际值。

v := reflect.ValueOf(&myVar)
if v.Kind() == reflect.Ptr {
elem := v.Elem() // 获取指针指向的值
}- 只有
Ptr类型的reflect.Value才能调用Elem()。 - 返回的是指向的目标值的
reflect.Value。 - 如果你尝试对非指针类型调用
Elem(),程序会 panic。 -
Elem()返回的值可以是结构体、基本类型、数组等,取决于原始指针指向的内容。
举个例子:
i := 42 p := &i v := reflect.ValueOf(p) elem := v.Elem() // 现在 elem 是 int 类型的 Value,值为 42 fmt.Println(elem.Int()) // 输出 42
Addr() 是用来做什么的?什么时候用?
有时候你手上有一个普通的值,比如结构体或者基本类型,但你想通过反射修改它的值,或者把它作为指针传递给某个函数。这时候你可以使用 Addr() 方法,获取该值的地址。
v := reflect.ValueOf(myStruct) ptr := v.Addr().Interface().(*MyStruct) // 转成具体指针类型
-
Addr()返回一个新的reflect.Value,表示原值的地址。 - 原始值必须是可寻址的(addressable),否则调用
Addr()会 panic。 - 常用于将非指针值转为指针,方便后续修改或传参。
注意:不是所有值都能调用 Addr()。例如从函数返回的临时值、切片元素、映射值等都不是 addressable 的。
常见错误:
s := MyStruct{}
v := reflect.ValueOf(s)
v.Addr() // panic: unaddressable value解决办法是把变量取地址再反射:
s := MyStruct{}
v := reflect.ValueOf(&s).Elem() // 先取地址,再解引用得到 addressable 的 struct
v.CanSet() // trueElem() 和 Addr() 的区别与配合使用
这两个方法看起来有点像反向操作,但在实际使用中它们有不同的职责:
-
Addr()是从一个值获取指针。 -
Elem()是从一个指针获取值。
它们常常一起出现,尤其是在反射中动态构造结构体指针并赋值的时候。
例如:
type S struct {
Name string
}
s := S{}
v := reflect.ValueOf(&s).Elem() // 获取结构体的可修改 Value
field := v.FieldByName("Name")
if field.IsValid() && field.CanSet() {
field.SetString("hello")
}在这个过程中:
- 用
&s得到指针; - 用
.Elem()得到结构体本身; - 接下来就可以操作字段了。
如果不加 .Elem(),那 reflect.Value 就是指针类型,不能直接设置字段值。
总结一下使用技巧
- 如果你拿到的是指针,想操作目标值,就用
Elem()。 - 如果你拿到的是普通值,但需要指针,就用
Addr()。 - 两者都不能乱用,否则容易 panic。
- 判断 Kind 是否为
Ptr,以及判断值是否 addressable,都是安全使用的前提。
基本上就这些。反射中的指针处理虽然不算复杂,但很容易忽略这些细节,导致运行时错误。










