首页 > 其他分享 >Golang反射获取变量类型和值

Golang反射获取变量类型和值

时间:2023-04-08 15:15:10浏览次数:44  
标签:反射 Kind 变量 fmt Elem Golang reflect TypeOf Printf

Golang反射获取变量类型和值

 

1. 什么是反射

反射是程序在运行期间获取变量的类型和值、或者执行变量的方法的能力。

Golang反射包中有两对非常重要的函数和类型,两个函数分别是:

reflect.TypeOf能获取类型信息reflect.Type

reflect.ValueOf 能获取数据的运行时表示reflect.Value

 

2. reflect.Type

Golang是一门静态类型的语言,反射是建立在类型之上的。

通过reflect.TypeOf()函数可以获得任意值的类型信息。

 

2.1 类型Type和种类Kind

诸如int32slicemap以及通过type关键词自定义的类型。

种类Kind可以理解为类型的具体分类。如int32type MyInt32 int32是两种不同类型,但都属于int32这个种类。

使用 reflect.TypeOf()获取变量类型以及种类。

  func main() {
  type MyInt32 int32
  a := MyInt32(1)
  b := int32(1)
  fmt.Printf("reflect.TypeOf(a):%v Kind:%v\n", reflect.TypeOf(a), reflect.TypeOf(a).Kind())
  fmt.Printf("reflect.TypeOf(b):%v Kind:%v\n", reflect.TypeOf(b), reflect.TypeOf(b).Kind())
  }

代码输出如下,由此可以看出int32type MyInt32 int32是两种不同类型,但都属于int32这个种类。

  $ go run main.go
  reflect.TypeOf(a):main.MyInt32 Kind:int32
  reflect.TypeOf(b):int32 Kind:int32
种类定义点击查看

 

2.2 引用指向元素的类型

// Elem returns a type's element type.
// It panics if the type's Kind is not Array, Chan, Map, Pointer, or Slice.
Elem() Type

部分情况我们需要获取指针指向元素的类型、或者slice元素的类型,可以reflect.Elem()函数获取。

  func main() {
  type myStruct struct {
  }
  a := &myStruct{}
  typeA := reflect.TypeOf(a)
  fmt.Printf("TypeOf(a):%v Kind:%v\n", typeA, typeA.Kind())
  fmt.Printf("TypeOf(a).Elem():%v Elem().Kind:%v\n", typeA.Elem(), typeA.Elem().Kind())
  s := []int64{}
  typeS := reflect.TypeOf(s)
  fmt.Printf("TypeOf(s):%v Kind:%v\n", typeS, typeS.Kind())
  fmt.Printf("TypeOf(s).Elem():%v Elem().Kind:%v\n", typeS.Elem(), typeS.Elem().Kind())
  }

代码输出如下,由此可以看出,通过reflect.Elem()函数可以获取引用指向数据的类型。

  $ go run main.go
  TypeOf(a):*main.myStruct Kind:ptr
  TypeOf(a).Elem():main.myStruct Elem().Kind:struct
  TypeOf(s):[]int64 Kind:slice
  TypeOf(s).Elem():int64 Elem().Kind:int64

 

2.3 结构体成员类型

通过NumField获取成员数量,Field通过下标访问成员的类型信息StructField,包括成员名称、类型、Tag信息等。

  func main() {
  type secStruct struct {
  Cnt []int64
  }
  type myStruct struct {
  Num int `json:"num_json" orm:"column:num_orm"`
  Desc string `json:"desc_json" orm:"column:desc_orm"`
  Child secStruct
  }
  s := myStruct{}
  typeS := reflect.TypeOf(s)
  // 成员数量
  fmt.Printf("NumField:%v \n", typeS.NumField())
  // 每个成员的信息 包括名称、类型、Tag
  for i := 0; i < typeS.NumField(); i++ {
  // 通过下标访问成员
  fmt.Printf("Field(%v):%+v\n", i, typeS.Field(i))
  }
  // 通过名称访问成员
  field, ok := typeS.FieldByName("Num")
  fmt.Printf("FieldByName(\"Num\") ok:%v field:%+v\n", ok, field)
  // 获取tag值
  fmt.Printf("json tag val:%+v\n", field.Tag.Get("json"))
  // 获取嵌套结构体的字段
  fmt.Printf("Cnt field:%+v\n", typeS.FieldByIndex([]int{2, 0}))
  }

代码输出如下,

  $ go run main.go
  NumField:3
  Field(0):{Name:Num PkgPath: Type:int Tag:json:"num_json" orm:"column:num_orm" Offset:0 Index:[0] Anonymous:false}
  Field(1):{Name:Desc PkgPath: Type:string Tag:json:"desc_json" orm:"column:desc_orm" Offset:8 Index:[1] Anonymous:false}
  Field(2):{Name:Child PkgPath: Type:main.secStruct Tag: Offset:24 Index:[2] Anonymous:false}
  FieldByName("Num") ok:true field:{Name:Num PkgPath: Type:int Tag:json:"num_json" orm:"column:num_orm" Offset:0 Index:[0] Anonymous:false}
  json tag val:num_json
  Cnt field:{Name:Cnt PkgPath: Type:[]int64 Tag: Offset:0 Index:[0] Anonymous:false}

 

3. reflect.Value

通过reflect.ValueOf获取变量值、值类型,种类为ArrayChanMapSlice, 或String可通过Len()获取长度

  func main() {
  b := int32(1)
  valueB := reflect.ValueOf(b)
  fmt.Printf("reflect.TypeOf(b):%v Kind:%v\n", valueB, valueB.Kind())
  s := "abcdefg"
  valueS := reflect.ValueOf(s)
  fmt.Printf("reflect.TypeOf(s):%v Kind:%v Len:%v\n", valueS, valueS.Kind(), valueS.Len())
  }

代码输出如下,

  $ go run main.go
  reflect.TypeOf(b):1 Kind:int32
  reflect.TypeOf(s):abcdefg Kind:string Len:7

 

3.1 结构体的成员的值

3.3 结构体成员类型获取结构体成员类型类似,reflect提供了NumField获取成员数量,Field通过下标访问成员的值。

  func main() {
  type secStruct struct {
  Cnt []int64
  }
  type myStruct struct {
  Num int `json:"num_json" orm:"column:num_orm"`
  Desc string `json:"desc_json" orm:"column:desc_orm"`
  Child secStruct
  }
  s := myStruct{
  Num: 100,
  Desc: "desc",
  Child: secStruct{[]int64{1, 2, 3}},
  }
  valueS := reflect.ValueOf(s)
  // 成员数量
  fmt.Printf("NumField:%v \n", valueS.NumField())
  // 每个成员的值
  for i := 0; i < valueS.NumField(); i++ {
  // 通过下标访问成员
  fmt.Printf("value(%v):%+v\n", i, valueS.Field(i))
  }
  // 通过名称访问成员
  value := valueS.FieldByName("Num")
  fmt.Printf("FieldByName(\"Num\") value:%v\n", value)
  // 获取嵌套结构体的字段
  fmt.Printf("Cnt field:%+v\n", valueS.FieldByIndex([]int{2, 0}))
  }

代码输出如下

  $ go run main.go
  NumField:3
  value(0):100
  value(1):desc
  value(2):{Cnt:[1 2 3]}
  FieldByName("Num") value:100
  Cnt field:[1 2 3]

 

3.2 遍历array、slice

通过func (v Value) Index(i int) Value可以通过下标来访问ArraySlice,或者 String各个元素的值。

  func main() {
  s := []int64{1, 2, 3, 4, 5, 6}
  valueS := reflect.ValueOf(s)
  fmt.Printf("ValueOf(s):%v Kind:%v Len:%v\n", valueS, valueS.Kind(), valueS.Len())
  for i := 0; i < valueS.Len(); i++ {
  fmt.Printf("valueS.Index(%v):%v\n", i, valueS.Index(i))
  }
  }

代码输出如下

  $ go run main.go
  ValueOf(s):[1 2 3 4 5 6] Kind:slice Len:6
  valueS.Index(0):1
  valueS.Index(1):2
  valueS.Index(2):3
  valueS.Index(3):4
  valueS.Index(4):5
  valueS.Index(5):6

 

3.3 遍历map

reflect有两种方法遍历map

  • 通过迭代器MapIter遍历map
  • 先获取map的所有key,再通过key获取对应的value
  func main() {
  m := map[int]string{
  1: "1",
  2: "2",
  3: "3",
  }
  valueM := reflect.ValueOf(m)
  // 迭代器访问
  iter := valueM.MapRange()
  for iter.Next() {
  fmt.Printf("key:%v val:%v\n", iter.Key(), iter.Value())
  }
  fmt.Println("------")
  // 通过key访问
  keys := valueM.MapKeys()
  for i := 0; i < len(keys); i++ {
  fmt.Printf("key:%v val:%v\n", keys[i], valueM.MapIndex(keys[i]))
  }
  }

代码输出如下,

  $ go run main.go
  key:1 val:1
  key:2 val:2
  key:3 val:3
  ------
  key:3 val:3
  key:1 val:1
  key:2 val:2

 

4. 反射的三大定律

反射的两个基础函数定义,

  • 获取类型func TypeOf(i any) Type
  • 获取值func ValueOf(i any) Value

其中,anyinterface{}的别名。

interface{}是不包含任何方法签名的空接口,任何类型都实现了空接口。

A value of interface type can hold any value that implements those methods.

因此,interface{}可以承载任何变量的 (value, concrete type)信息。

 

4.1 从interface到反射对象

interface{}承载变量的(value, concrete type)信息,通过反射暴露方法来访问interface{}的值和类型。

可以简单理解为interface{}的值和信息传递到reflect.Type和 reflect.Value,方便获取。

 

4.2 从反射对象到interface

可以通过函数func (v Value) Interface() (i any)将反射对象转换为interface{}

func ValueOf(i any) Value的反向操作。

  func main() {
  a := int32(10)
  valueA := reflect.ValueOf(a)
  fmt.Printf("ValueOf(a):%v\n", valueA)
  fmt.Printf("Interface():%v\n", valueA.Interface())
  ai, ok := valueA.Interface().(int32)
  fmt.Printf("ok:%v val:%v\n", ok, ai)
  }

代码输出如下

  $ go run main.go
  ValueOf(a):10
  Interface():10
  ok:true val:10

 

4.3 通过反射修改对象,该对象值必须是可修改的

reflect提供func (v Value) CanSet() bool判断对象值是否修改,通过func (v Value) Set(x Value)修改对象值

  func main() {
  a := int32(10)
  valueA := reflect.ValueOf(a)
  fmt.Printf("valueA :%v\n", valueA.CanSet())
  b := int32(100)
  valuePtrB := reflect.ValueOf(&b)
  fmt.Printf("valuePtrB:%v Elem:%v\n", valuePtrB.CanSet(), valuePtrB.Elem().CanSet())
  valuePtrB.Elem().Set(reflect.ValueOf(int32(200)))
  fmt.Printf("b:%v Elem:%v\n", b, valuePtrB.Elem())
  }

 代码输出如下

  $ go run main.go
  valueA :false
  valuePtrB:false Elem:true
  b:200 Elem:200

后续章节再分享通过修改各种类型的值的实操。

 

5. 参考文档

laws-of-reflection

interface

标签:反射,Kind,变量,fmt,Elem,Golang,reflect,TypeOf,Printf
From: https://www.cnblogs.com/golandhome/p/17298565.html

相关文章

  • 使用Golang实现Nginx代理功能
    由于业务需要实现对多个web应用做同域二级目录代理,用NGINX的又感觉太重了,而且不好做配置页面,用golang来实现代理功能支持正则表达式匹配机制支持多应用多级目录代理。支持应用子路由代理支持webapi代理支持websocket代理支持禁用缓存设置支持http、https混合使用支持/dir......
  • golang中的substring
    发现golangstrings包中没有javastring.substring方法,于是尝试了下:想要在java中实现提取“世”的需求//java代码Strings="Hello,世界";System.out.println(s.substring(7,8));//世在golang中尝试这样://golang代码s1:="Hello,世界"fmt.Println(s1[7:8])//�因为......
  • 巧如范金,精比琢玉,一分钟高效打造精美详实的Go语言技术简历(Golang1.18)
    转自刘悦研发少闲月,九月人倍忙。又到了一年一度的“金九银十”秋招季,又到了写简历的时节,如果你还在用传统的Word文档寻找模板,然后默默耕耘,显然就有些落后于时代了,本次我们尝试使用云平台flowcv高效打造一份巧如范金、精比琢玉的高品质Golang技术简历。首先来到云平台:flowcv.com......
  • 注解与反射
    注解注解概念注解在java中一般都是@xxx例如@Override就是覆盖注解可以在包,类,方法等上面通过反射方式访问内置注解@SupressWarnings:把警告弄没元注解注解一般需要参数,直接ctrl点进去查看有什么参数选择Method,运行在方法上面@Target就是规定注解的范围为//测试元......
  • 定义Person类包含两个成员变量,结构体构造方法---Java
    packagepracticeHomework;/**定义一个Person类,含有一个String类型的成员变量和一个int型成员变量*分别为这两变量定义访问方法和修改方法*再为该类定义一个speak的方法,输出name和age的值**编写程序,使用Person类,实现数据的访问的修改*/publicclassPerson{......
  • 《渗透测试》安全开发-PHP应用&留言板功能&超全局变量&数据库操作&第三方插件引用 202
    PHPStorm:专业PHP开发IDEPhpStudy:ApacheMYSQL环境NavicatPremium:全能数据库管理工具 #数据导入-mysql架构&库表列1、数据库名,数据库表名,数据库列名2、数据库数据,格式类型,长度,键等 #数据库操作-mysqli函数&增删改查PHP函数:连接,选择,执行,结果,关闭等参考:https://......
  • ReflectASM高性能反射Java库
    1.简介  在日常开发中,经常需要使用反射操作类中的属性或方法。ReflectASM是一个非常小的Java库,它提供高性能反射能力。  Github地址:https://github.com/EsotericSoftware/reflectasm2.官方性能报告3.用法ReflectASM仅可访问public属性和方法。pom引入依赖<de......
  • flask-day5——python项目高并发异步部署、uwsgi启动python的web项目不要使用全局变量
    目录一、python项目高并发异步部署二、uwsgi启动Python的Web项目中不要使用全局变量三、信号3.1flask信号3.2django信号四、微服务的概念五、flask-script六、sqlalchemy快速使用七、sqlalchemy快速使用4.1原生操作的快速使用八、创建操作数据表九、作业1、什么是猴子补丁,有什......
  • 编写一个程序以确定分别由signed及unsigned限定的char、short、int及long类型变量的取
    #include<stdio.h>#include<limits.h>main(){printf("采用打印标准头文件limits.h的相应值\n");//signertypesprintf("signedcharmin=%d\n",SCHAR_MIN);printf("signedcharmax=%d\n",SCHAR_MAX);prin......
  • golang TLS方式发送邮件
    packagemailimport( "crypto/tls" "errors" "fmt" "net/smtp" "net/textproto")typeloginAuthstruct{ username,passwordstring}//LoginAuthisfuncLoginAuth(usernamestring,passwordstring)......