Go REFLECT Library | 01 - 反射的类型 Type

简介: Go REFLECT Library | 01 - 反射的类型 Type

一、反射概述

反射是指程序在运行期间对程序本身进行访问和修改的能力。程序在编译过程中变量会被转换为内存地址,变量名不会被编译器写入到可执行部分。在程序运行时程序无法获取自身的信息。

在静态语言中如 Java 可以在程序编译期将变量的反射信息,如字段名称、类型等信息整合到可执行文件中,并给程序提供接口访问反射信息,这样就可以在程序运行期获取类型的反射信息,并修改该它们。

对于动态语言来说如 Ruby 的动态特性相比静态语言来说可以非常简单的在程序运行时访问变量、方法或者对象信息,也可以修改它们,甚至可以动态性可以让程序自己构造并执行代码,这就是元编程。

Ruby 中的基类(Object)包含了方法 methods、常量 constants 和实例变量instance_variable 的动态获取。

puts String.method_defined?(:upcase) # 判断是否定义了 upcase 方法
puts String.methods # 获取所有方法
puts Math.const_get("PI") # 获取常量
puts Math.const_set("PII", 1000) # 设置常量
puts Math.const_defined?(:P) # 判断是否包含指定常量
puts Math.constants # 获取所有常量
复制代码

因此 Ruby 这里动态解释型语言是没有反射系统的,但是 Go 作为一门静态编译型语言提供了 reflect 标准库访问程序的反射信息。

Go 语言的反射系统无法获取到一个可执行文件空间中或者是一个包中所有类型信息,需要配合使用标准库中对应的词法和语法解析器和抽象语法书对源码进行扫描后获取这些信息

image.png

二、反射类型对象

基本数类型的 反射类型对象

在 Go 中使用 reflect 标准库下的 TypeOf 函数可以获取任意变量的反射类型对象,程序通过 反射类型对象 可以访问任意变量的类型信息。

func main(){
   zulu := "stark"
   zuluType := reflect.TypeOf(zulu)
   fmt.Printf("zuluType 的类型为:%v,类型名为:%v,种类为:%v\n", zuluType, zuluType.Name(), zuluType.Kind())
}
复制代码

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

zuluType 的类型为:string,类型名为:string,种类为:string
复制代码

TypeOf 函数返回一个 Type 接口,该接口包含非常多的方法

image.png

上述代码中的类型就是变量的数据类型,如基本数据类型中的 int、int64、float64、string、map、bool 以及 type 结构体类型等,类型名就是类型本身。

image.png

种类既 Kind 方法获取的信息是指对象归属的品种,在 reflect 库中对对象归属的 Kind 做了定义

image.png

Kind 的范围在如下列出的常量中

image.png

并在通过 String() 方法做了小写的转换,最终返回 Kind 为 string

image.png

Name 和 Kind 可以表示一个变量的 反射类型对象 的信息。每种数据类型变量的 反射类型对象 的 Name 和 Kind 都是不同的。

引用数据类型的 反射类型对象

func main(){
   zulu := map[string]string{
      "name": "Stark",
      "address": "NYC",
   }
   zuluType := reflect.TypeOf(zulu)
   fmt.Printf("zuluType 的类型为:%v,类型名为:%v,种类为:%v\n", zuluType, zuluType.Name(), zuluType.Kind())
}
复制代码

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

zuluType 的类型为:map[string]string,类型名为:,种类为:map
复制代码

Map、Array、Slice 和 Pointer 类型的 Name() 都为空字符串

结构体的 反射类型对象

func main(){
   zulu := Zulu{"stark", 33}
   zuluType := reflect.TypeOf(zulu)
   fmt.Printf("zuluType 的类型为:%v,类型名为:%v,种类为:%v\n", zuluType, zuluType.Name(), zuluType.Kind())
}
type Zulu struct {
   Name string
   Age int
}
复制代码

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

zuluType 的类型为:main.Zulu,类型名为:Zulu,种类为:struct
复制代码

结构体变量的 反射类型对象 的 Name 就是结构体的名字,种类为 struct 结构体

指针的 反射类型对象

func main(){
   zulu := Zulu{"stark", 33}
   // 定义一个指针
   zuluPtr := &zulu
   zuluType := reflect.TypeOf(zuluPtr)
   fmt.Printf("zuluType 的类型为:%v,类型名为:%v,种类为:%v\n", zuluType, zuluType.Name(), zuluType.Kind())
}
type Zulu struct {
   Name string
   Age int
}
复制代码

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

zuluType 的类型为:*main.Zulu,类型名为:,种类为:ptr
复制代码

指针的 Name() 返回的也是空字符串。

在 main 函数中增加代码

// 其余代码保持不变,在 main 函数底部增加如下代码。
// 使用反射类型对象(Type)获取原类型
zuluTypeElem := zuluType.Elem()
fmt.Printf("zuluTypeElem 的类型为:%v,类型名为:%v,种类为:%v\n", zuluTypeElem, zuluTypeElem.Name(), zuluTypeElem.Kind())
复制代码

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

zuluType 的类型为:*main.Zulu,类型名为:,种类为:ptr
zuluTypeElem 的类型为:main.Zulu,类型名为:Zulu,种类为:struct
复制代码

也就是说我们通过一个结构体指针获取了一个反射类型,在通过反射类型获取到原结构体

image.png

Go 中对指针获取 反射类型对象 之后,可以通过获取的 反射类型对象Elem 方法获取指针所执行的元素的类型,这个过程被称为取元素,就相当于对指针执行了 * 操作。


相关文章
|
6天前
|
JSON 人工智能 Go
go 反射的常见用法
go 反射的常见用法
15 4
|
6天前
|
存储 人工智能 Go
Go type assertions
Go type assertions
16 3
|
6天前
|
人工智能 JSON NoSQL
Go MongoDB Driver 中的 A D M E 类型是什么
Go MongoDB Driver 中的 A D M E 类型是什么
13 1
|
6天前
|
JSON 人工智能 编译器
Go json 能否解码到一个 interface 类型的值
Go json 能否解码到一个 interface 类型的值
11 1
|
6天前
|
存储 人工智能 Java
深入理解 go reflect - 要不要传指针
深入理解 go reflect - 要不要传指针
5 0
|
6天前
|
存储 缓存 人工智能
深入理解 go reflect - 反射为什么慢
深入理解 go reflect - 反射为什么慢
5 0
|
6天前
|
存储 人工智能 JSON
深入理解 go reflect - 反射基本原理
深入理解 go reflect - 反射基本原理
13 0
|
8天前
|
JSON 中间件 Go
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
本文详细介绍了如何在Go项目中集成并配置Zap日志库。首先通过`go get -u go.uber.org/zap`命令安装Zap,接着展示了`Logger`与`Sugared Logger`两种日志记录器的基本用法。随后深入探讨了Zap的高级配置,包括如何将日志输出至文件、调整时间格式、记录调用者信息以及日志分割等。最后,文章演示了如何在gin框架中集成Zap,通过自定义中间件实现了日志记录和异常恢复功能。通过这些步骤,读者可以掌握Zap在实际项目中的应用与定制方法
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
|
1天前
|
安全 Java Go
探索Go语言在高并发环境中的优势
在当今的技术环境中,高并发处理能力成为评估编程语言性能的关键因素之一。Go语言(Golang),作为Google开发的一种编程语言,以其独特的并发处理模型和高效的性能赢得了广泛关注。本文将深入探讨Go语言在高并发环境中的优势,尤其是其goroutine和channel机制如何简化并发编程,提升系统的响应速度和稳定性。通过具体的案例分析和性能对比,本文揭示了Go语言在实际应用中的高效性,并为开发者在选择合适技术栈时提供参考。
|
5天前
|
运维 Kubernetes Go
"解锁K8s二开新姿势!client-go:你不可不知的Go语言神器,让Kubernetes集群管理如虎添翼,秒变运维大神!"
【8月更文挑战第14天】随着云原生技术的发展,Kubernetes (K8s) 成为容器编排的首选。client-go作为K8s的官方Go语言客户端库,通过封装RESTful API,使开发者能便捷地管理集群资源,如Pods和服务。本文介绍client-go基本概念、使用方法及自定义操作。涵盖ClientSet、DynamicClient等客户端实现,以及lister、informer等组件,通过示例展示如何列出集群中的所有Pods。client-go的强大功能助力高效开发和运维。
24 1