Go语言 反射

简介: 1. reflect 包1.1 获取变量类型1.2 断言处理类型转换2. ValueOf2.1 获取变量值2.2 类型转换3. Value.Set3.1 设置变量值3.2 示例

引言


反射指的是运行时动态的获取变量的相关信息


1. reflect 包


类型是变量,类别是常量

reflect.TypeOf,获取变量的类型,返回reflect.Type类型
reflect.ValueOf,获取变量的值,返回reflect.Value类型
reflect.Value.Kind,获取变量的类别,返回一个常量
reflect.Value.Interface(),转换成interface{}类型


在这里插入图片描述


1.1 获取变量类型


package main
import (
  "fmt"
  "reflect"
)
func Test(i interface{}) {
  //反射数据类型
  t := reflect.TypeOf(i)
  fmt.Println("类型是", t)
  //反射数据值
  v := reflect.ValueOf(i)
  fmt.Println("值是", v)
}
func main() {
  a := "hello"
  Test(a)
}
//输出结果如下
类型是 string
值是 hello
package main
import (
  "fmt"
  "reflect"
)
type Student struct {
  Name  string
  Age   int
  Score float32
}
func Test(i interface{}) {
  //反射获取类型
  t := reflect.TypeOf(i)
  fmt.Println("类型是", t)
  //反射获取值
  v := reflect.ValueOf(i)
  //判断值的类别
  c := v.Kind()
  fmt.Println("类别是", c)
}
func main() {
  var stu Student = Student{
    Name:  "张三",
    Age:   18,
    Score: 80,
  }
  Test(stu)
  fmt.Println("-------------")
  var num int = 10
  Test(num)
}
//输出结果如下
类型是 main.Student
类别是 struct
-------------
类型是 int
类别是 int


1.2 断言处理类型转换


package main
import (
  "fmt"
  "reflect"
)
type Student struct {
  Name  string
  Age   int
  Score float32
}
func Test(i interface{}) {
  t := reflect.TypeOf(i)
  fmt.Println("类型是", t)
  //类别
  v := reflect.ValueOf(i)
  c := v.Kind()
  fmt.Println("类别是", c)
  fmt.Printf("c的类型是%T\n", c)
  fmt.Printf("v的类型是%T\n", v)
  //转换成接口
  iv := v.Interface()
  fmt.Printf("iv的类型%T\n", iv)
  //断言处理
  stu_iv, err := iv.(Student)
  if err {
    fmt.Printf("stu_iv的类型%T\n", stu_iv)
  }
}
func main() {
  var stu Student = Student{
    Name:  "张三",
    Age:   18,
    Score: 80,
  }
  Test(stu)
}
//输出结果如下
类型是 main.Student
类别是 struct
c的类型是reflect.Kind
v的类型是reflect.Value
iv的类型main.Student
stu_iv的类型main.Student


2. ValueOf


2.1 获取变量值


reflect.valueof(x).Float()
reflect.valueof(x).Int()
reflect.valueof(x).String()
reflect.Valueof(x).Bool()


2.2 类型转换


package main
import (
  "fmt"
  "reflect"
)
func Test(i interface{}) {
  v := reflect.ValueOf(i)
  fmt.Printf("v的类型是%T\n", v)
  //转换成指定类型
  t := v.Int()
  fmt.Printf("t的类型是%T\n", t)
}
func main() {
  //类型不同的话会报错
  var num int = 100
  Test(num)
}
//输出结果如下
v的类型是reflect.Value
t的类型是int64


3. Value.Set


3.1 设置变量值


reflect.Value.SetFloat(),设置浮点数
reflect.value.SetInt(),设置整数
reflect.Value.SetString(),设置字符串


3.2 示例


package main
import (
  "fmt"
  "reflect"
)
func Test(i interface{}) {
  v := reflect.ValueOf(i)
  //更新值需要value的地址,否则会保存,Elem()表示指针*
  v.Elem().SetInt(100)
  result := v.Elem().Int()
  fmt.Printf("result类型为 %T, 值为 %d\n", result, result)
}
func main() {
  var num int = 10
  Test(&num)
}
//输出结果如下
result类型为 int64, 值为 100


4. 结构体反射


反射出结构体的属性和方法数量

方法名需大写,需要被跨包调用识别


4.1 查看结构体字段数量和方法数量


package main
import (
  "fmt"
  "reflect"
)
//结构体
type Student struct {
  Name  string
  Age   int
  Score float32
}
//结构体方法
func (s Student) Run() {
  fmt.Println("Running")
}
func (s Student) Sleep() {
  fmt.Println("Sleeping")
}
//使用反射查看结构体字段数量和方法数量
func Test(i interface{}) {
  v := reflect.ValueOf(i)
  //类别判断
  if v.Kind() != reflect.Struct {
    fmt.Println("不是结构体")
    return
  }
  //获取结构体中字段数量
  stu_num := v.NumField()
  fmt.Println("字段数量: ", stu_num)
  //获取结构体中方法数量
  stu_meth := v.NumMethod()
  fmt.Println("方法数量: ", stu_meth)
}
func main() {
  var stu Student = Student{
    Name:  "张三",
    Age:   18,
    Score: 88,
  }
  Test(stu)
}
//输出结果如下
字段数量:  3
方法数量:  2


4.2 获取结构体属性


package main
import (
  "fmt"
  "reflect"
)
type Student struct {
  Name  string
  Age   int
  Score float32
}
func Test(i interface{}) {
  v := reflect.ValueOf(i)
  //获取结构体中每个属性
  for i := 0; i < v.NumField(); i++ {
    //输出属性值
    fmt.Printf("%d %v\n", i, v.Field(i))
    //输出属性值的类型
    fmt.Printf("%d %v\n", i, v.Field(i).Kind())
  }
}
func main() {
  var stu Student = Student{
    Name:  "张三",
    Age:   18,
    Score: 88,
  }
  Test(stu)
}
//输出结果如下
0 张三
0 string
1 18
1 int
2 88
2 float32


4.3 更改属性值


package main
import (
  "fmt"
  "reflect"
)
type Student struct {
  Name  string
  Age   int
  Score float32
}
func Test(i interface{}, name string) {
  v := reflect.ValueOf(i)
  vk := v.Kind()
  //判断是都为指针并指向结构体类型
  if vk != reflect.Ptr && v.Elem().Kind() == reflect.Struct {
    fmt.Println("expect struct")
    return
  }
  //更改属性值
  v.Elem().Field(0).SetString(name)
  //获取结构体中每个属性
  for i := 0; i < v.Elem().NumField(); i++ {
    //输出属性值
    fmt.Printf("%d %v\n", i, v.Elem().Field(i))
    //输出属性值的类型
    fmt.Printf("%d %v\n", i, v.Elem().Field(i).Kind())
  }
}
func main() {
  var stu Student = Student{
    Name:  "张三",
    Age:   18,
    Score: 88,
  }
  Test(&stu, "李四")
}
//输出结果如下
0 李四
0 string
1 18
1 int
2 88
2 float32


4.4 Tag原信息处理


package main
import (
  "encoding/json"
  "fmt"
  "reflect"
)
type Student struct {
  Name  string `json:"stu_name"`
  Age   int
  Score float32
}
func Test(i interface{}, name string) {
  v := reflect.ValueOf(i)
  vk := v.Kind()
  //判断是都为指针并指向结构体类型
  if vk != reflect.Ptr && v.Elem().Kind() == reflect.Struct {
    fmt.Println("expect struct")
    return
  }
  //更改属性值
  v.Elem().Field(0).SetString(name)
  //获取结构体中每个属性
  for i := 0; i < v.Elem().NumField(); i++ {
    //输出属性值
    fmt.Printf("%d %v\n", i, v.Elem().Field(i))
    //输出属性值的类型
    fmt.Printf("%d %v\n", i, v.Elem().Field(i).Kind())
  }
}
func main() {
  var stu Student = Student{
    Name:  "张三",
    Age:   18,
    Score: 88,
  }
  Test(&stu, "李四")
  fmt.Println("----------------json原信息----------------")
  result, _ := json.Marshal(stu)
  fmt.Println("json原信息: ", string(result))
  //反射获取类型
  st := reflect.TypeOf(stu)
  s := st.Field(0)
  fmt.Printf("Name原信息名称: %s\n", s.Tag.Get("json"))
}
//输出结果如下
0 李四
0 string
1 18
1 int
2 88
2 float32
----------------json原信息----------------
json原信息:  {"stu_name":"李四","Age":18,"Score":88}
Name原信息名称: stu_name


5. 函数反射


Go 中函数是可以赋值给变量的


  • 示例:

既然函数可以像普通的类型变量一样,那么在反射机制中就和不同的变量是一样的,在反射中函数和方法的类型(Type)都是reflect.Func,如果要调用函数,通过 ValueCall() 方法

package main
import (
  "fmt"
  "reflect"
)
func hello() {
  fmt.Println("hello world")
}
func main() {
  //反射使用函数
  v := reflect.ValueOf(hello)
  //类型判断是否属于reflect.func类型
  if v.Kind() == reflect.Func {
    fmt.Println("函数")
  }
  //反射调用函数
  v.Call(nil)   //Call中需要传入的是切片
}
//输出结果如下
函数
hello world
package main
import (
  "fmt"
  "reflect"
  "strconv"
)
//反射调用传参和返回值函数
func Test(i int) string {
  return strconv.Itoa(i)
}
func main() {
  v := reflect.ValueOf(Test)
  //定义参数切片
  params := make([]reflect.Value, 1)
  //切片元素赋值
  params[0] = reflect.ValueOf(20)
  //反射调函数
  result := v.Call(params)
  fmt.Printf("result的类型是 %T\n", result)
  //[]reflect.Value切片转换string
  s := result[0].Interface().(string)
  fmt.Printf("s的类型是 %T ,值为 %s\n", s, s)
}
//输出结果如下
result的类型是 []reflect.Value
s的类型是 string ,值为 20


6. 方法反射


反射中方法的调用,函数和方法可以说其实本质上是相同的,只不过方法与一个“对象”进行了“绑定”,方法是“对象”的一种行为,这种行为是对于这个“对象”的一系列操作,例如修改“对象”的某个属性


6.1 使用 MethodByName 名称调用方法


package main
import (
  "fmt"
  "reflect"
  "strconv"
)
//反射方法
type Student struct {
  Name string
  Age  int
}
//结构体方法
func (s *Student) SetName(name string) {
  s.Name = name
}
func (s *Student) SetAge(age int) {
  s.Age = age
}
func (s *Student) String() string {
  return fmt.Sprintf("%p", s) + ",Name:" + s.Name + ",Age:" + strconv.Itoa(s.Age)
}
func main() {
  //实例化
  stu := &Student{"张三", 19}
  //反射获取值:指针方式
  stuV := reflect.ValueOf(&stu).Elem()
  fmt.Println("修改前: ", stuV.MethodByName("String").Call(nil)[0])
  //修改值
  params := make([]reflect.Value, 1)    //定义切片
  params[0] = reflect.ValueOf("李四")
  stuV.MethodByName("SetName").Call(params)
  params[0] = reflect.ValueOf(20)
  stuV.MethodByName("SetAge").Call(params)
  fmt.Println("修改后: ", stuV.MethodByName("String").Call(nil)[0])
}
//输出结果如下
修改前:  0xc000004078,Name:张三,Age:19
修改后:  0xc000004078,Name:李四,Age:20


6.2 使用 method 索引调用方法


package main
import (
  "fmt"
  "reflect"
  "strconv"
)
//反射方法
type Student struct {
  Name string
  Age  int
}
//结构体方法
func (s *Student) B(name string) {
  s.Name = name
}
func (s *Student) A(age int) {
  s.Age = age
}
func (s *Student) C() string {
  return fmt.Sprintf("%p", s) + ",Name:" + s.Name + ",Age:" + strconv.Itoa(s.Age)
}
func main() {
  //实例化
  stu := &Student{"张三", 19}
  //反射获取值:指针方式
  stuV := reflect.ValueOf(&stu).Elem()
  //索引调用方法
  fmt.Println("修改前: ", stuV.Method(2).Call(nil)[0])
  params := make([]reflect.Value, 1)
  params[0] = reflect.ValueOf("李四")
  stuV.Method(1).Call(params)
  params[0] = reflect.ValueOf(20)
  stuV.Method(0).Call(params)
  fmt.Println("修改后: ", stuV.Method(2).Call(nil)[0])
  //调用索引大小取决于方法名称的ASCII大小进行排序
}
//输出结果如下
修改前:  0xc000004078,Name:张三,Age:19
修改后:  0xc000004078,Name:李四,Age:20
相关文章
|
3天前
|
运维 监控 算法
监控局域网其他电脑:Go 语言迪杰斯特拉算法的高效应用
在信息化时代,监控局域网成为网络管理与安全防护的关键需求。本文探讨了迪杰斯特拉(Dijkstra)算法在监控局域网中的应用,通过计算最短路径优化数据传输和故障检测。文中提供了使用Go语言实现的代码例程,展示了如何高效地进行网络监控,确保局域网的稳定运行和数据安全。迪杰斯特拉算法能减少传输延迟和带宽消耗,及时发现并处理网络故障,适用于复杂网络环境下的管理和维护。
|
4天前
|
编译器 Go
揭秘 Go 语言中空结构体的强大用法
Go 语言中的空结构体 `struct{}` 不包含任何字段,不占用内存空间。它在实际编程中有多种典型用法:1) 结合 map 实现集合(set)类型;2) 与 channel 搭配用于信号通知;3) 申请超大容量的 Slice 和 Array 以节省内存;4) 作为接口实现时明确表示不关注值。此外,需要注意的是,空结构体作为字段时可能会因内存对齐原因占用额外空间。建议将空结构体放在外层结构体的第一个字段以优化内存使用。
|
8天前
|
存储 Go
Go 语言入门指南:切片
Golang中的切片(Slice)是基于数组的动态序列,支持变长操作。它由指针、长度和容量三部分组成,底层引用一个连续的数组片段。切片提供灵活的增减元素功能,语法形式为`[]T`,其中T为元素类型。相比固定长度的数组,切片更常用,允许动态调整大小,并且多个切片可以共享同一底层数组。通过内置的`make`函数可创建指定长度和容量的切片。需要注意的是,切片不能直接比较,只能与`nil`比较,且空切片的长度为0。
Go 语言入门指南:切片
|
12天前
|
算法 安全 Go
公司局域网管理系统里的 Go 语言 Bloom Filter 算法,太值得深挖了
本文探讨了如何利用 Go 语言中的 Bloom Filter 算法提升公司局域网管理系统的性能。Bloom Filter 是一种高效的空间节省型数据结构,适用于快速判断元素是否存在于集合中。文中通过具体代码示例展示了如何在 Go 中实现 Bloom Filter,并应用于局域网的 IP 访问控制,显著提高系统响应速度和安全性。随着网络规模扩大和技术进步,持续优化算法和结合其他安全技术将是企业维持网络竞争力的关键。
26 2
公司局域网管理系统里的 Go 语言 Bloom Filter 算法,太值得深挖了
|
4天前
|
存储 缓存 监控
企业监控软件中 Go 语言哈希表算法的应用研究与分析
在数字化时代,企业监控软件对企业的稳定运营至关重要。哈希表(散列表)作为高效的数据结构,广泛应用于企业监控中,如设备状态管理、数据分类和缓存机制。Go 语言中的 map 实现了哈希表,能快速处理海量监控数据,确保实时准确反映设备状态,提升系统性能,助力企业实现智能化管理。
22 3
|
8天前
|
开发框架 前端开发 Go
eino — 基于go语言的大模型应用开发框架(二)
本文介绍了如何使用Eino框架实现一个基本的LLM(大语言模型)应用。Eino中的`ChatModel`接口提供了与不同大模型服务(如OpenAI、Ollama等)交互的统一方式,支持生成完整响应、流式响应和绑定工具等功能。`Generate`方法用于生成完整的模型响应,`Stream`方法以流式方式返回结果,`BindTools`方法为模型绑定工具。此外,还介绍了通过`Option`模式配置模型参数及模板功能,支持基于前端和用户自定义的角色及Prompt。目前主要聚焦于`ChatModel`的`Generate`方法,后续将继续深入学习。
102 7
|
4天前
|
存储 缓存 安全
Go 语言中的 Sync.Map 详解:并发安全的 Map 实现
`sync.Map` 是 Go 语言中用于并发安全操作的 Map 实现,适用于读多写少的场景。它通过两个底层 Map(`read` 和 `dirty`)实现读写分离,提供高效的读性能。主要方法包括 `Store`、`Load`、`Delete` 等。在大量写入时性能可能下降,需谨慎选择使用场景。
|
8天前
|
存储 开发框架 Devops
eino — 基于go语言的大模型应用开发框架(一)
Eino 是一个受开源社区优秀LLM应用开发框架(如LangChain和LlamaIndex)启发的Go语言框架,强调简洁性、可扩展性和可靠性。它提供了易于复用的组件、强大的编排框架、简洁明了的API、最佳实践集合及实用的DevOps工具,支持快速构建和部署LLM应用。Eino不仅兼容多种模型库(如OpenAI、Ollama、Ark),还提供详细的官方文档和活跃的社区支持,便于开发者上手使用。
74 8
|
18天前
|
监控 Linux PHP
【02】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-2月12日优雅草简化Centos stream8安装zabbix7教程-本搭建教程非docker搭建教程-优雅草solution
【02】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-2月12日优雅草简化Centos stream8安装zabbix7教程-本搭建教程非docker搭建教程-优雅草solution
68 20
|
8天前
|
存储 算法 Go
Go语言实战:错误处理和panic_recover之自定义错误类型
本文深入探讨了Go语言中的错误处理和panic/recover机制,涵盖错误处理的基本概念、自定义错误类型的定义、panic和recover的工作原理及应用场景。通过具体代码示例介绍了如何定义自定义错误类型、检查和处理错误值,并使用panic和recover处理运行时错误。文章还讨论了错误处理在实际开发中的应用,如网络编程、文件操作和并发编程,并推荐了一些学习资源。最后展望了未来Go语言在错误处理方面的优化方向。

热门文章

最新文章