Go REFLECT Library | 06 - reflect.Type 和 reflect.Value 应用

简介: Go REFLECT Library | 06 - reflect.Type 和 reflect.Value 应用

一、使用 reflect.Type 创建实例

在通过 reflect.TypeOf 函数获取到变量的反射类型对象之后,可以通过反射类型对象 reflect.TypeNew 函数来创建一个新的实例,注意这个实例的类型是 reflect.Type 类型的。

package main
import (
   "fmt"
   "reflect"
)
func main() {
   var zulu int
   zuluTypeOf := reflect.TypeOf(zulu)
   // 创建类型实例
   zuluIntPtr := reflect.New(zuluTypeOf)
   fmt.Printf("%T\n", zuluIntPtr)
   fmt.Printf("%v, %v\n", zuluIntPtr.Type(), zuluIntPtr.Kind())
}
复制代码

执行上述代码,输出结果如下:

reflect.Value
*int, ptr
复制代码

二、使用 reflect.Value 调用函数

Go 编程 | 连载 15 - Go 语言的函数 讲到 Go 中函数是一等公民,函数可以保存在变量中,当 reflect.ValueOf 获取的是一个函数类型变量的反射值对象时,可以通过 reflect.ValueCall 方法调用该函数

使用反射调用函数需要将参数使用反射值对象的切片 []reflect.Value 构造后传入 Call 方法中,调用之后再通过切片 []reflect.Value 的形式返回函数的返回值。

函数只有一个返回值的情况

定义一个 add 函数,该函数有两个入参和一个返回值,先获取函数类型变量的反射值对象,再利用反射值对象调用 Call 函数,并将函数变量中函数的入参以 reflect.Value 类型切片的形式作为 Call 函数的入参即可获取函数变量中函数调用的结果。

package main
import (
   "fmt"
   "reflect"
)
func main() {
   // 定义一个变量保存 add 函数
   var victor func(x, y int) (sum int) = add
   // 获取 函数变量的反射值对象
   victorValueOf := reflect.ValueOf(victor)
   // 构造 []reflect.Value 切片参数列表
   paramValueOfSlice := []reflect.Value{reflect.ValueOf(3), reflect.ValueOf(8)}
   // 调用函数
   var resValueOfSlice []reflect.Value = victorValueOf.Call(paramValueOfSlice)
   // 遍历
   fmt.Printf("%v\n", resValueOfSlice[0])
   fmt.Printf("%T\n", resValueOfSlice[0])
}
func add(x, y int) (sum int) {
   sum = x + y
   return
}
复制代码

执行上述代码,输出结果如下:

11
reflect.Value
复制代码

函数有两个或者多个返回值的情况

定义一个 div 函数,该函数有两个入参和两个返回值,仍然是先获取函数类型变量的反射值对象,再利用反射值对象调用 Call 函数,并将函数变量中函数的入参以 reflect.Value 类型切片的形式作为 Call 函数的入参即可获取函数变量中函数调用的结果。

该函数调用结果的切片中含有两个元素,且这两个元素的类型都是 reflect.Value

package main
import (
   "fmt"
   "reflect"
)
func main() {
   // 定义一个变量保存 add 函数
   var victor func(x, y int) (sum int, ok bool) = div
   // 获取 函数变量的反射值对象
   victorValueOf := reflect.ValueOf(victor)
   fmt.Println("当除数非 0 的情况:")
   // 构造 []reflect.Value 切片参数列表
   paramValueOfSlice := []reflect.Value{reflect.ValueOf(8), reflect.ValueOf(2)}
   // 调用函数
   var resValueOfSlice []reflect.Value = victorValueOf.Call(paramValueOfSlice)
   // 返回值切片长度
   fmt.Printf("返回值切片长度为:%v\n", len(resValueOfSlice))
   for i := 0; i < len(resValueOfSlice); i++ {
      fmt.Printf("%v, %T\n", resValueOfSlice[i], resValueOfSlice[i])
   }
   fmt.Println()
   fmt.Println("当除数为 0 的情况:")
   // 构造 []reflect.Value 切片参数列表
   paramValueOfSlice2 := []reflect.Value{reflect.ValueOf(3), reflect.ValueOf(0)}
   // 调用函数
   var resValueOfSlice2 []reflect.Value = victorValueOf.Call(paramValueOfSlice2)
   // 返回值切片长度
   fmt.Printf("返回值切片长度为:%v\n", len(resValueOfSlice2))
   for i := 0; i < len(resValueOfSlice2); i++ {
      fmt.Printf("%v, %T\n", resValueOfSlice2[i], resValueOfSlice2[i])
   }
}
func div(x, y int) (res int, ok bool) {
   if y == 0 {
      res = 0
      ok = false
   } else {
      res = x / y
      ok = true
   }
   return
}
复制代码

执行上述代码,输出结果如下:

返回值切片长度为:2
4, reflect.Value
true, reflect.Value
当除数为 0 的情况:
返回值切片长度为:2
0, reflect.Value
false, reflect.Value
复制代码

不管是函数只有一个返回值还是多个返回值的情况,使用反射值对象调用 Call 函数获取到的函数变量中函数的调用结果都是 reflect.Value 类型的,要想获取原类型的数据,还需要将 reflect.Value 再转换成原类型。


相关文章
|
12天前
|
存储 缓存 NoSQL
【Go语言专栏】Go语言中的Redis操作与缓存应用
【4月更文挑战第30天】本文探讨了在Go语言中使用Redis进行操作和缓存应用的方法。文章介绍了Redis作为高性能键值存储系统,用于提升应用性能。推荐使用`go-redis/redis`库,示例代码展示了连接、设置、获取和删除键值对的基本操作。文章还详细阐述了缓存应用的步骤及常见缓存策略,包括缓存穿透、缓存击穿和缓存雪崩的解决方案。利用Redis和合适策略可有效优化应用性能。
|
6天前
|
负载均衡 监控 Go
Golang深入浅出之-Go语言中的服务网格(Service Mesh)原理与应用
【5月更文挑战第5天】服务网格是处理服务间通信的基础设施层,常由数据平面(代理,如Envoy)和控制平面(管理配置)组成。本文讨论了服务发现、负载均衡和追踪等常见问题及其解决方案,并展示了使用Go语言实现Envoy sidecar配置的例子,强调Go语言在构建服务网格中的优势。服务网格能提升微服务的管理和可观测性,正确应对问题能构建更健壮的分布式系统。
27 1
|
9天前
|
安全 Go
Golang深入浅出之-Go语言中的并发安全队列:实现与应用
【5月更文挑战第3天】本文探讨了Go语言中的并发安全队列,它是构建高性能并发系统的基础。文章介绍了两种实现方法:1) 使用`sync.Mutex`保护的简单队列,通过加锁解锁确保数据一致性;2) 使用通道(Channel)实现无锁队列,天生并发安全。同时,文中列举了并发编程中常见的死锁、数据竞争和通道阻塞问题,并给出了避免这些问题的策略,如明确锁边界、使用带缓冲通道、优雅处理关闭以及利用Go标准库。
24 5
|
10天前
|
JSON 监控 安全
Golang深入浅出之-Go语言中的反射(reflect):原理与实战应用
【5月更文挑战第1天】Go语言的反射允许运行时检查和修改结构,主要通过`reflect`包的`Type`和`Value`实现。然而,滥用反射可能导致代码复杂和性能下降。要安全使用,应注意避免过度使用,始终进行类型检查,并尊重封装。反射的应用包括动态接口实现、JSON序列化和元编程。理解反射原理并谨慎使用是关键,应尽量保持代码静态类型。
22 2
|
11天前
|
Go
go 反射Reflect
go 反射Reflect
|
12天前
|
存储 缓存 监控
【Go语言专栏】Go语言应用的性能调优实践
【4月更文挑战第30天】本文介绍了Go语言应用的性能调优技巧,包括使用`pprof`进行性能分析、选择正确算法与数据结构、减少内存分配、优化并发及避免阻塞操作、选用合适锁机制。此外,文章还提到了编译选项如`-trimpath`和`-ldflags`,以及系统资源和环境调优。通过实例展示了代码优化、并发处理和锁的使用。最后,推荐了进一步学习资源,鼓励读者深入探索Go语言的性能优化。
|
12天前
|
存储 NoSQL Go
【Go语言专栏】Go语言中的MongoDB操作与NoSQL应用
【4月更文挑战第30天】本文介绍了Go语言中操作MongoDB的方法和NoSQL应用的优势。MongoDB作为流行的NoSQL数据库,以其文档型数据模型、高性能和可扩展性被广泛应用。在Go语言中,通过mongo-go-driver库可轻松实现与MongoDB的连接及插入、查询、更新和删除等操作。MongoDB在NoSQL应用中的优点包括灵活的数据模型、高性能、高可用性和易于扩展,使其成为处理大规模数据和高并发场景的理想选择。
|
12天前
|
缓存 监控 前端开发
【Go 语言专栏】Go 语言中的 WebSocket 实时通信应用
【4月更文挑战第30天】本文探讨了Go语言在WebSocket实时通信中的应用。WebSocket作为全双工通信协议,允许持续的双向通信。Go语言凭借其高效和并发特性,适合构建实时应用。文中概述了在Go中实现WebSocket的基本步骤,包括服务器和客户端的建立与通信,并列举了实时聊天、数据监控和在线协作等应用案例。同时,强调了消息格式、并发处理、错误处理和安全性的注意事项。通过数据压缩、缓存管理和连接管理等策略可优化性能。Go语言还能与数据库和前端框架结合,提升用户体验。总之,Go语言为WebSocket实时通信提供了强大支持,有望在更多领域发挥作用。
|
NoSQL Java 测试技术
Go应用单元测试实践
Go应用单元测试搭建
Go应用单元测试实践
|
14小时前
|
存储 编译器 Go
Go语言学习12-数据的使用
【5月更文挑战第5天】本篇 Huazie 向大家介绍 Go 语言数据的使用,包含赋值语句、常量与变量、可比性与有序性
16 6
Go语言学习12-数据的使用