Go语言学习教程之反射的示例详解

目录
  • 介绍
  • 反射的规律
    • 1. 从接口值到反射对象的反射
    • 2. 从反射对象到接口值的反射
    • 3. 要修改反射对象,该值一定是可设置的

介绍

reflect包实现运行时反射,允许一个程序操作任何类型的对象。典型的使用是:取静态类型interface{}的值,通过调用TypeOf获取它的动态类型信息,调用ValueOf会返回一个表示运行时数据的一个值。本文通过记录对reflect包的简单使用,来对反射有一定的了解。本文使用的Go版本:

$ go version
go version go1.18 darwin/amd64

在了解反射之前,先了解以下几个概念:

  • 静态类型:每个变量都有一个静态类型,这个类型是在编译时(compile time)就已知且固定的。
  • 动态类型:接口类型的变量还有一个动态类型,是在运行时(run time)分配给变量的值的一个非接口类型。(除非分配给变量的值是nil,因为nil没有类型)。
  • 空接口类型interface{} (别名any)表示空的方法集,它可以是任何值的类型,因为任何值都满足有0或多个方法(有0个方法一定是任何值的子集)。
  • 一个接口类型的变量存储一对内容:分配给变量的具体的值,以及该值的类型描述符。可以示意性地表示为(value, type)对,这里的type是具体的类型,而不是接口类型。

反射的规律

1. 从接口值到反射对象的反射

在基本层面,反射只是检测存储在接口变量中的(value, type)对的一种机制。

可以使用reflect包的 reflect.ValueOfreflect.TypeOf方法,获取接口变量值中的(value, type)对,类型分别为reflect.Valuereflect.Type

(1)TypeOf方法:

func TypeOf(i any) Type

TypeOf返回表示i的动态类型的反射Type。如果inil,那么返回nil

(2)ValueOf方法:

func ValueOf(i any) Value

ValueOf返回一个新的Value,初始化为存储在接口i中的具体值。ValueOf(nil) 会返回零Value。这个零Value是反射对象中表示没有值的Value

var a interface{} = 1
var b interface{} = 1.11
var c string = "aaa"

// 将接口类型的变量运行时存储的具体的值和类型显示地获取到
fmt.Println("type:", reflect.TypeOf(a))   // type: int
fmt.Println("value:", reflect.ValueOf(a)) // value: 1

fmt.Println("type:", reflect.TypeOf(nil))   // type: <nil>
fmt.Println("value:", reflect.ValueOf(nil)) // value: <invalid reflect.Value>

fmt.Println("type:", reflect.TypeOf(b))   // type: float64
fmt.Println("value:", reflect.ValueOf(b)) // value: 1.11

fmt.Println("type:", reflect.TypeOf(c))   // type: string
fmt.Println("value:", reflect.ValueOf(c)) // value: aaa

reflect.Value 的 Type方法,返回一个 reflect.Value的类型。

reflect.ValueString方法,将reflect.Value的底层值作为字符串返回。

fmt.Printf使用了反射:

  • %T使用reflect.TypeOf,拿到变量的动态类型。
  • %v深入到 reflect.Value内部拿到变量具体的值。
var a interface{} = 1

fmt.Println("type:", reflect.ValueOf(a).Type())     // type: int
fmt.Println("string:", reflect.ValueOf(a).String()) // string: <int Value>
fmt.Printf("type: %T \n", a)                        // type: int
fmt.Printf("string: %v \n", a)                      // string: 1

Type 和 Value都有一个 Kind 方法,返回一个表示存储的项的类型的常量。 比如UintFloat64Slice等。

Value的类似IntFloat这种名称的方法能够获取存储在内部的值。

Value 的“getter”(取值) 和 “setter”(设置值)会对能保存该值的最大类型进行操作。比如对于所有有符号整数,都是int64。

var a interface{} = 1
var b interface{} = 1.11

reflectA := reflect.ValueOf(a)
fmt.Println("kind: ", reflectA.Kind()) // kind:  int

reflectIntA := reflectA.Int()               // 返回的是 能存储有符号整数的最大类型 的值
reflectFloatB := reflect.ValueOf(b).Float() // 返回的是 能存储浮点数的最大类型 的值

var a1 int64 = reflectIntA
var b1 float64 = reflectFloatB

// var a2 int32 = reflectIntA // 会报错:cannot use reflectIntA (variable of type int64) as int32 value in variable
// var b2 float32 = reflectFloatB // 会报错:cannot use reflectFloatB (variable of type float64) as float32 value in variable
fmt.Println("a1: ", a1, "b1: ", b1) // a1:  1 b1:  1.11

2. 从反射对象到接口值的反射

像物理反射一样,Go 中的反射产生了它自己的逆。可以使用reflect.ValueInterface方法还原一个接口值。Interface() 将类型和值信息打包回一个接口表示。

Interface方法:

func (v Value) Interface() (i any)

Interfacev的当前值作为一个interface{}返回。它等同于:

var i interface{} = (v 的底层值)

练习代码:

var a interface{} = 1

reflectA := reflect.ValueOf(a)

var a3 interface{} = reflectA.Interface()
var a4 int = reflectA.Interface().(int) // 使用接口值的类型断言

fmt.Println(a3, a4) // 1 1 ,因为fmt.Println接收接口类型interface{}的参数,使用 reflect.Value 拿到具体的值,所以打印出运行时的具体结果

3. 要修改反射对象,该值一定是可设置的

可设置性是reflect.Value的一个属性,表示一个反射对象可以修改用于创建该反射对象的实际存储。可设置性可以通过CanSet方法获得。如果对不可设置的reflect.Value调用Set方法,就会报错。

使用reflect.Value 类型的Elem方法能通过指针间接寻址得到一个可设置性为真的reflect.Value

var d float64 = 2.222

fmt.Println(reflect.ValueOf(d).CanSet()) // false

reflectD := reflect.ValueOf(&d).Elem()
fmt.Println(reflectD.CanSet()) // true
reflectD.SetFloat(3.33)

fmt.Println(d, reflectD) // 3.33 3.33

reflect.ValueOf(d) 是通过复制d中的内容得到的reflect.Value类型的值,它复制的内容存放的内存地址 和d的值存放的内存地址是不同的。所以不能通过它来修改d中原本存储的内容。

上面的代码中可以看到,调用SetFloat(3.33)之后,反射对象reflectD和创建该反射对象的d都发生了改变。

使用反射修改结构体的字段:

type T struct {
    A int
    B string
}
t := T{111, "xxx"}
s := reflect.ValueOf(&t).Elem()

typeOfT := s.Type()
for i := 0; i < s.NumField(); i++ { // NumField 返回结构体中的字段数量
    f := s.Field(i)
    fmt.Printf("%d: %s %s = %v\n",
        i,
        typeOfT.Field(i).Name, // 获取第i个字段的名称
        f.Type(),              // 获取第i个字段的类型
        f.Interface(),         // 将第i个字段转换回接口类型的值
    )
    // 0: A int = 111
    // 1: B string = xxx
}

s.Field(0).SetInt(222)      // 设置结构体的第一个字段的值
s.Field(1).SetString("yyy") // 设置结构体的第二个字段的值
fmt.Println(t)              // {222 yyy}

上述练习代码都在一个reflect.go文件中,练习时在终端执行go run reflect.go运行该文件。

以上就是Go语言学习教程之反射的示例详解的详细内容,更多关于Go语言 反射的资料请关注我们其它相关文章!

(0)

相关推荐

  • Go系列教程之反射的用法

    反射是 Go 语言的高级主题之一.我会尽可能让它变得简单易懂. 本教程分为如下小节. 什么是反射? 为何需要检查变量,确定变量的类型? reflect 包 reflect.Type 和 reflect.Value reflect.Kind NumField() 和 Field() 方法 Int() 和 String() 方法 完整的程序 我们应该使用反射吗? 让我们来逐个讨论这些章节. 什么是反射? 反射就是程序能够在运行时检查变量和值,求出它们的类型.你可能还不太懂,这没关系.在本教程结束后,

  • Go语言基础反射示例详解

    目录 概述 语法 一.基本操作 二.修改目标对象 三.动态调用方法 总结 示例 概述 在程序运行期对程序动态的进行访问和修改 reflect godoc: https://golang.org/pkg/reflect/ reflect包有两个数据类型: Type:数据类型 [reflect.TypeOf():是获取Type的方法] Value:值的类型[reflect.ValueOf():是获取Value的方法] 语法 一.基本操作 获取变量类型 func TypeOf(i interface{

  • Go语言的反射机制详解

    反射是语言里面是非常重要的一个特性,我们经常会看见这个词,但是对于反射没有一个很好的理解,主要是因为对于反射的使用场景不太熟悉. 一.理解变量的内在机制 1.类型信息,元信息,是预先定义好的,静态的. 2.值信息,程序进行过程中,动态变化的. 二.反射和空接口 1.空接口相当于一个容器,能接受任何东西. 2.那怎么判断空接口变量存储的是什么类型呢?之前有使用过类型断言,这只是一个比较基础的方法 3.如果想获取存储变量的类型信息和值信息就要使用反射机制,所以反射是什么? 反射就是动态的获取变量类型

  • 图文详解go语言反射实现原理

    Go反射的实现和 interface 和 unsafe.Pointer 密切相关.如果对golang的 interface 底层实现还没有理解,可以去看我之前的文章: Go语言interface底层实现 , unsafe.Pointer 会在后续的文章中做介绍. (本文目前使用的Go环境是Go 1.12.9) interface回顾 首先我们简单的回顾一下interface的结构,总体上是: 细分下来分为有函数的 iface 和无函数的 eface (就是 interface{} ); 无函数的

  • Go语言学习之反射的用法详解

    目录 1. reflect 包 1.1 获取变量类型 1.2 断言处理类型转换 2. ValueOf 2.1 获取变量值 2.2 类型转换 3. Value.Set 3.1 设置变量值 3.2 示例 4. 结构体反射 4.1 查看结构体字段数量和方法数量 4.2 获取结构体属性 4.3 更改属性值 4.4 Tag原信息处理 5. 函数反射 6. 方法反射 6.1 使用 MethodByName 名称调用方法 6.2 使用 method 索引调用方法 反射指的是运行时动态的获取变量的相关信息 1.

  • 学习使用Go反射的用法示例

    什么是反射 大多数时候,Go中的变量,类型和函数非常简单直接.当需要一个类型.变量或者是函数时,可以直接定义它们: type Foo struct { A int B string } var x Foo func DoSomething(f Foo) { fmt.Println(f.A, f.B) } 但是有时你希望在运行时使用变量的在编写程序时还不存在的信息.比如你正在尝试将文件或网络请求中的数据映射到变量中.或者你想构建一个适用于不同类型的工具.在这种情况下,你需要使用反射.反射使您能够在

  • Go语言学习教程之反射的示例详解

    目录 介绍 反射的规律 1. 从接口值到反射对象的反射 2. 从反射对象到接口值的反射 3. 要修改反射对象,该值一定是可设置的 介绍 reflect包实现运行时反射,允许一个程序操作任何类型的对象.典型的使用是:取静态类型interface{}的值,通过调用TypeOf获取它的动态类型信息,调用ValueOf会返回一个表示运行时数据的一个值.本文通过记录对reflect包的简单使用,来对反射有一定的了解.本文使用的Go版本: $ go version go version go1.18 dar

  • Struts2学习教程之输入校验示例详解

    前言 数据校验几乎是每个应用都要做的工作.用户输入的数据,发送到服务器端,天知道用户输入的数据是否是合法的,是否为恶意输入.所以一个健壮的应用系统必须对用户的输入进行校验,将非法的输入阻止在应用之外,防止这些非法的输入进入系统,从而保证系统的稳定性.安全性. 我们都知道,为了更好的用户体验,以及更高的效率,现在的Web应用都存在以下两重数据校验: 客户端数据校验 服务器端数据校验 对于客户端数据校验主要是通过JavaScript代码来完成:而对于服务器端数据校验是整个应用阻止非法数据的最后防线,

  • R语言学习ggplot2绘制统计图形包全面详解

    目录 一.序 二.ggplot2是什么? 三.ggplot2能画出什么样的图? 四.组装机器 五.设计图纸 六.机器的零件 1. 零件--散点图 1) 变换颜色 2) 拟合曲线 3) 变换大小 4) 修改透明度 5) 分层 6) 改中文 2. 零件--直方图与条形图 1) 直方图 2) 润色 3) 条形图 3. 零件--饼图 4. 零件--箱线图 5. 零件--小提琴图 6. 零件打磨 7. 超级变变变 8. 其他常用零件 七.实践出真知 八.学习资源 九.参考资料 一.序 作为一枚统计专业的学

  • Java中的反射机制示例详解

    目录 反射 什么是Class类 获取Class实例的三种方式 通过反射创建类对象 通过反射获取类属性.方法.构造器 更改访问权限和实例赋值 运用场景 反射 反射就是把Java类中的各个成分映射成一个个的Java对象.即在运行状态中,对于任意一个类,都能够知道这个类的所以属性和方法:对于任意一个对象,都能调用它的任意一个方法和属性.这种动态获取信息及动态调用对象方法的功能叫Java的反射机制 每一个Java程序执行必须通过编译.加载.链接和初始化四个阶段 1.编译:将.java.文件编译成字节码.

  • Go语言学习之context包的用法详解

    目录 前言 需求一 需求二 Context 接口 emptyCtx valueCtx 类型定义 WithValue cancelCtx 类型定义 cancelCtx WithCancel timerCtx 类型定义 WithDeadline WithTimeout 总结 前言 日常 Go 开发中,Context 包是用的最多的一个了,几乎所有函数的第一个参数都是 ctx,那么我们为什么要传递 Context 呢,Context 又有哪些用法,底层实现是如何呢?相信你也一定会有探索的欲望,那么就跟

  • Go语言中的字符串处理方法示例详解

    1 概述 字符串,string,一串固定长度的字符连接起来的字符集合.Go语言的字符串是使用UTF-8编码的.UTF-8是Unicode的实现方式之一. Go语言原生支持字符串.使用双引号("")或反引号(``)定义. 双引号:"", 用于单行字符串. 反引号:``,用于定义多行字符串,内部会原样解析. 示例: // 单行 "心有猛虎,细嗅蔷薇" // 多行 ` 大风歌 大风起兮云飞扬. 威加海内兮归故乡. 安得猛士兮守四方! ` 字符串支持转义

  • Mysql教程分组排名实现示例详解

    目录 1.数据源 2.数据整体排名 1)普通排名 2)并列排名 3)并列排名 3.数据分组后组内排名 1)分组普通排名 2)分组后并列排名 3)分组后并列排名 4.分组后取各组的前两名 1.数据源 2.数据整体排名 1)普通排名 从1开始,按照顺序一次往下排(相同的值也是不同的排名). set @rank =0; select city , score, @rank := @rank+1 rank from cs order by score desc; 结果如下: 2)并列排名 相同的值是相同

  • TensorFlow人工智能学习数据合并分割统计示例详解

    目录 一.数据合并与分割 1.tf.concat() 2.tf.split() 3.tf.stack() 二.数据统计 1.tf.norm() 2.reduce_min/max/mean() 3.tf.argmax/argmin() 4.tf.equal() 5.tf.unique() 一.数据合并与分割 1.tf.concat() 填入两个tensor, 指定某维度,在指定的维度合并.除了合并的维度之外,其他的维度必须相等. 2.tf.split() 填入tensor,指定维度,指定分割的数量

  • 人工智能学习Pytorch梯度下降优化示例详解

    目录 一.激活函数 1.Sigmoid函数 2.Tanh函数 3.ReLU函数 二.损失函数及求导 1.autograd.grad 2.loss.backward() 3.softmax及其求导 三.链式法则 1.单层感知机梯度 2. 多输出感知机梯度 3. 中间有隐藏层的求导 4.多层感知机的反向传播 四.优化举例 一.激活函数 1.Sigmoid函数 函数图像以及表达式如下: 通过该函数,可以将输入的负无穷到正无穷的输入压缩到0-1之间.在x=0的时候,输出0.5 通过PyTorch实现方式

  • Go语言基础设计模式之策略模式示例详解

    目录 概述 针对同一类型问题的多种处理方式 一.不使用策略模式 二.策略模式 UML 总结 示例 概述 定义一系列算法,将每个算法封装起来.并让它们能够相互替换.策略模式让算法独立于使用它的客户而变化. 针对同一类型问题的多种处理方式 一.不使用策略模式 package main import "fmt" type User struct { Name string } func (this User) travel(t string) { switch t { case "

随机推荐