一文初探Go语言中的reflect反射包
作者:陈明勇 发布时间:2024-02-07 01:31:51
reflect 反射包
针对反射,Go
提供了 reflect
包,使用这个包里的函数可以在程序运行时获取和更新未知变量的值,操作未知变量的方法等。
reflect
包核心的两个重要类型:
reflect.Type
:Type
是一个接口,不同数据类型有着不同的结构体实现。这个接口用于操作变量的类型信息,类型的信息只能读取。reflect.Value
:Value
是一个结构体,通过这个结构体可以操作变量的值。
TypeOf(i) 和 ValueOf(i)
reflect.TypeOf(i any) Type
:获取变量的类型,返回一个reflect.Type
类型。reflect.ValueOf(i any) Value
:获取变量的值,返回reflect.Value
类型,通过Value
可以对获取变量更多的信息。
案例1:获取变量的类别和类型信息
import (
"fmt"
"reflect"
)
type User struct {
Name string
}
func main() {
user := User{
Name: "cmy",
}
func4Reflect(user)
}
func func4Reflect(data any) {
typ := reflect.TypeOf(data)
fmt.Println("类别:", typ.Kind()) // 类别: struct
fmt.Println("类型:", typ.Name()) // 类型: User
}
通过
TypeOf()
函数获取data
的类型信息,然后调用Kind()
和Name()
方法分别获取data
变量的类别和类型信息。根据返回结果可知,
Kind()
返回的是Go
的数据类型,而Name()
返回的是我们自定义的数据类型。根据
Kind()
返回值的特点,可以用于判断变量属于 Go 的哪种数据类型,用于类型限制等场景。
案例2:修改基本数据类型变量的值
import (
"fmt"
"reflect"
)
func main() {
num1 := 666
fmt.Println("num1 原值:", num1)
func4Reflect(&num1)
fmt.Println("num1 修改后的值:", num1)
num2 := 0.5
fmt.Println("num2 原值:", num2)
func4Reflect(&num2)
fmt.Println("num2 修改后的值:", num2)
str := "go"
fmt.Println("str 原值:", str)
func4Reflect(&str)
fmt.Println("str 修改后的值:", str)
}
func func4Reflect(data any) {
typ := reflect.TypeOf(data)
val := reflect.ValueOf(data)
switch typ.Elem().Kind() {
case reflect.Int:
val.Elem().SetInt(888)
case reflect.Float64:
val.Elem().SetFloat(3.14)
case reflect.String:
val.Elem().SetString("Golang")
}
}
通过 ValueOf()
函数获取 data
变量的值信息,然后结合 reflect.Type.Kind()
方法,对不同类型的变量的值进行修改操作(只举三种类型的例子):
int
类型 → 使用SetInt(val)
方法对值进行修改。float64
→ 使用SetFloat(val)
方法对值进行修改。string
类型 → 使用SetString(val)
方法对值进行修改。
data
必须是指针类型,否则无法通过反射修改。
由于是指针类型,因此需要调用 Elem()
方法获取到指针指向的变量,才能修改变量的值。
案例3:通过反射获取结构体的字段名、字段类型和字段的值
import (
"fmt"
"reflect"
)
type User struct {
Name string
Age int
}
func main() {
user := User{
Name: "cmy",
Age: 18,
}
func4Reflect(user)
}
func func4Reflect(data any) {
typ := reflect.TypeOf(data)
val := reflect.ValueOf(data)
// 获取结构体字段的数量
numField := val.NumField()
for i := 0; i < numField; i++ {
fmt.Println("字段名称:", typ.Field(i).Name)
fmt.Println("字段类型:", typ.Field(i).Type.Name())
fmt.Println("字段值:", val.Field(i).Interface())
fmt.Println("----------------------------")
}
}
首先通过
TypeOf()
和ValueOf()
获取到结构体的类型信息和值信息。其次通过
Value.NumField()
方法获取到结构体字段的数量。接着遍历结构体的字段,通过
Type.Field(i)
方法,传入索引,获取到对应字段的类型信息,通过Name
属性获取字段名,Type.Name()
获取字段类型。最后通过
Value.Field(i)
方法,传入索引,获取到对应字段的值信息,通过Interface()
方法获取字段实际的值。
小结
本文首先介绍了 reflect
包里两个重要的类型 reflect.Type
和 reflect.Value
,简单说明了它们的作用;其次介绍了TypeOf(i)
和 ValueOf(i)
两个函数;最后通过三个案例介绍了它们的使用场景。
来源:https://juejin.cn/post/7173317932355846157
猜你喜欢
- 本文实例为大家分享了ADO.NET通用数据库访问类,供大家参考学习,具体内容如下using System;using System.Coll
- 本文实例讲述了Python设计模式之备忘录模式原理与用法。分享给大家供大家参考,具体如下:备忘录模式(Memento Pattern):不破
- 前言VScode是一个相当优秀的IDE,具备开源、跨平台、模块化、插件丰富、启动时间快、颜值高、可高度定制等等优秀的特质,不愧是微软爸爸的私
- 本文实例讲述了Python3.4解释器用法。分享给大家供大家参考,具体如下:Linux/Unix的系统上,Python解释器通常被安装在 /
- 遇到两次mysql密码忘记了?最开始干了最傻的事,卸载了重装。现在有一个不用卸载也能把密码设置回来的办法。知识来源于网络,我这里稍加整理,遇
- PDOStatement::nextRowsetPDOStatement::nextRowset — 在一个多行集语句句柄中推进到下一个行集
- 本文记录了mysql 8.0.14 安装配置的过程,供大家参考,具体内容如下1.下载地址:下载地址找到zip压缩文件.2.配置环境变量把解压
- 先看代码m = nn.Identity(54, unused_argument1=0.1, unused_argument2=False)i
- 如果 select 元素下的所有 option 元素均没有指定 selected 属性,会默认选中第一个。可以通过 select.selec
- FBV:function based view 基于函数的视图.CBV:class based view 基于类的视图.在视图函数创建类,需
- 思路:先选择在线签名网站,找到接口模拟请求,然后将生成的签名图片显示在 Tkinter 生成的 GUI 窗口上,最后保存生成的签名图片选择网
- 为了保证程序的健壮性与容错性,即在遇到错误时候程序不会崩溃,我们需要对异常进行处理,1.if进行处理,在错误发生之前进行预防如果错误发生的条
- MySQL的默认的调度策略可用总结如下:· 写入操作优先于读取操作。· 对某张数据表的写入操作某一时刻只能发生一次,写入请求按照它们到达的次
- 本文实例讲述了Python实现的自定义多线程多进程类。分享给大家供大家参考,具体如下:最近经常使用到对大量文件进行操作的程序以前每次写的时候
- <html><head><style>body{font-family:宋体;font-size:16p
- 概念django自带一套信号机制来帮助我们在框架的不同位置之间传递信息。也就是说,当某一事件发生时,信号系统可以允许一个或多个发送者(sen
- 说到转置操作,顺便提及矩阵与数组的区别:矩阵:数学里的概念,其元素只能是数值,这也是区别于数组的根本所在数组:计算机中的概念,代表一种数据组
- 已经记不得是在哪个网站上看到的了,一般情况下对于验证码的校验,大家很容易写成下面这样: <% If Request.Form(&quo
- 目录初始化程序创建Surface对象事件监听游戏循环Pygame 作为一个入门级的游戏开发库,其实并不难学,只要掌握 Python 编程的相
- 在日常的工作中,我们通常会有去探测目标主机是否存活的应用场景,单个的服务器主机可以通过计算机自带的DOS命令来执行,但是业务的存在往往不是单