golang reflect反射之Elem()方法

本文通过一个Go语言实例,详细解析了如何使用反射获取结构体类型,并解释了指针类型在反射中的表现形式及如何使用Elem方法获取指针背后的元素类型。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

实例代码:

type struct MsgData{}
func main() {

	msg := &MsgData{}
	funcTyp := reflect.TypeOf(msg)
	fmt.Println("funcTyp.Name = ", funcTyp.Name())
	fmt.Println("funcTyp.Kind = ", funcTyp.Kind())

	funcTyp = funcTyp.Elem()
	fmt.Println("funcTyp.Name = ", funcTyp.Name())
	fmt.Println("funcTyp.Kind = ", funcTyp.Kind())
}

输出结果:

funcTyp.Name =
funcTyp.Kind =  ptr    

funcTyp.Name =  MsgData
funcTyp.Kind =  struct 

可以看到 对于 Type类型的funcTyp,直接取Kind时,它的类型为ptr,也就是在反射中所有的指针类型都是ptr,但是当我们想要获取指针背后元素的真正数据类型时就需要使用Elem方法,Elem源码如下(省略了多个case,只关注Pointer):

func (t *rtype) Elem() Type {
	switch t.Kind() {
	省略。。。
	case Pointer:
		tt := (*ptrType)(unsafe.Pointer(t))
		return toType(tt.elem)
	省略。。。
	}
	panic("reflect: Elem of invalid type " + t.String())
}

可以看到它是把指针类型的保存的地址取出来,然后转换为Type类型。

### Golang Reflect `Elem` 方法详解 #### 定义与作用 在 Go 语言中,反射包提供了强大的工具用于操作未知类型的对象。对于指针类型而言,`reflect.Value.Elem()` 方法返回该指针对应的实际值的反射表示形式[^4]。 当面对一个指向某个具体类型的指针时,如果想要访问或修改其内部存储的数据,则需要先通过 `Elem()` 来解引用此指针得到真正的数据项。这使得即使是在运行期间也能安全有效地处理各种复杂的数据结构。 #### 使用场景举例 假设有一个整数类型的指针变量,在不知道确切类型的情况下如何对其进行读写?此时就可以借助于反射机制以及 `Elem()` 方法实现这一需求: ```go package main import ( "fmt" "reflect" ) func modifyValue(ptr interface{}) { v := reflect.ValueOf(ptr).Elem() if !v.CanSet() { // 检查是否允许设置新值 fmt.Println("cannot set value") return } switch v.Kind() { case reflect.Int: v.SetInt(999) default: fmt.Printf("unsupported type %s\n", v.Type()) } } func main() { var num int = 123 modifyValue(&num) fmt.Println(num) // 输出应该是被改变后的数值 999 } ``` 上述例子展示了怎样利用 `Elem()` 去间接地更改传入参数所指向的内容。这里特别注意的是要始终验证是否有权限去设定新的属性(`CanSet`),因为并非所有的反射值都能够支持这样的操作;只有那些可寻址且不是不可变的对象才具备这种能力。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值