一文初探 Go reflect 反射包

简介: 本文首先介绍了 reflect 包里两个重要的类型 reflect.Type 和 reflect.Value,简单说明了它们的作用;其次介绍了TypeOf(i) 和 ValueOf(i) 两个函数;最后通过三个案例介绍了它们的使用场景。

耐心和持久胜过激烈和狂热。

哈喽大家好,我是陈明勇,本文分享的知识是 Go reflect 包的基本使用。如果本文对你有帮助,不妨点个赞,如果你是 Go 语言初学者,不妨点个关注,一起成长一起进步,如果本文有错误的地方,欢迎指出!

reflect 反射包

针对反射,Go 提供了 reflect 包,使用这个包里的函数可以在程序运行时获取和更新未知变量的值,操作未知变量的方法等。

  • reflect 包核心的两个重要类型:
  • reflect.TypeType 是一个接口,不同数据类型有着不同的结构体实现。这个接口用于操作变量的类型信息,类型的信息只能读取。
  • reflect.ValueValue 是一个结构体,通过这个结构体可以操作变量的值。

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.Typereflect.Value,简单说明了它们的作用;其次介绍了TypeOf(i) 和  ValueOf(i) 两个函数;最后通过三个案例介绍了它们的使用场景。

目录
相关文章
|
9月前
|
Go 开发者
Go语言包的组织与导入 -《Go语言实战指南》
本章详细介绍了Go语言中的包(Package)概念及其使用方法。包是实现代码模块化、复用性和可维护性的核心单位,内容涵盖包的基本定义、命名规则、组织结构以及导入方式。通过示例说明了如何创建和调用包,并深入讲解了`go.mod`文件对包路径的管理。此外,还提供了多种导入技巧,如别名导入、匿名导入等,帮助开发者优化代码结构与可读性。最后以表格形式总结了关键点,便于快速回顾和应用。
370 61
|
5月前
|
Java 编译器 Go
【Golang】(1)Go的运行流程步骤与包的概念
初次上手Go语言!先来了解它的运行流程吧! 在Go中对包的概念又有怎样不同的见解呢?
292 4
|
8月前
|
JSON 中间件 Go
Go语言实战指南 —— Go中的反射机制:reflect 包使用
Go语言中的反射机制通过`reflect`包实现,允许程序在运行时动态检查变量类型、获取或设置值、调用方法等。它适用于初中级开发者深入理解Go的动态能力,帮助构建通用工具、中间件和ORM系统等。
415 63
|
7月前
|
缓存 监控 安全
告别缓存击穿!Go 语言中的防并发神器:singleflight 包深度解析
在高并发场景中,多个请求同时访问同一资源易导致缓存击穿、数据库压力过大。Go 语言提供的 `singleflight` 包可将相同 key 的请求合并,仅执行一次实际操作,其余请求共享结果,有效降低系统负载。本文详解其原理、实现及典型应用场景,并附示例代码,助你掌握高并发优化技巧。
498 0
|
10月前
|
Go 持续交付 开发者
Go语言包与模块(module)的基本使用-《Go语言实战指南》
本章深入讲解Go语言中的包(Package)和模块(Module)概念。包是代码组织的最小单位,每个`.go`文件属于一个包,通过`import`实现复用;主程序包需命名为`main`。模块是Go 1.11引入的依赖管理机制,支持自动版本管理和私有/远程仓库,无需依赖GOPATH。通过实际示例,如自定义包`mathutil`和第三方模块`gin`的引入,展示其使用方法。常用命令包括`go mod init`、`go mod tidy`等,帮助开发者高效管理项目依赖。最后总结,包负责功能划分,模块实现现代化依赖管理,提升团队协作效率。
402 15
|
Linux Go iOS开发
怎么禁用 vscode 中点击 go 包名时自动打开浏览器跳转到 pkg.go.dev
本文介绍了如何在 VSCode 中禁用点击 Go 包名时自动打开浏览器跳转到 pkg.go.dev 的功能。通过将 gopls 的 `ui.navigation.importShortcut` 设置为 &quot;Definition&quot;,可以实现仅跳转到定义处而不打开链接。具体操作步骤包括:打开设置、搜索 gopls、编辑 settings.json 文件并保存更改,最后重启 VSCode 使设置生效。
662 9
怎么禁用 vscode 中点击 go 包名时自动打开浏览器跳转到 pkg.go.dev
|
Go 数据库
Go语言中的包(package)是如何组织的?
在Go语言中,包是代码组织和管理的基本单元,用于集合相关函数、类型和变量,便于复用和维护。包通过目录结构、文件命名、初始化函数(`init`)及导出规则来管理命名空间和依赖关系。合理的包组织能提高代码的可读性、可维护性和可复用性,减少耦合度。例如,`stringutils`包提供字符串处理函数,主程序导入使用这些函数,使代码结构清晰易懂。
435 11
|
Go 索引
go语言使用strings包
go语言使用strings包
341 3
|
编译器 Go 开发者
go语言中导入相关包
【11月更文挑战第1天】
219 3
Go to Learn Go之反射
Go to Learn Go之反射
161 8