- 使用 Spring Initializr 创建 Spring Boot 应用程序
- 在Spring Boot中配置Cassandra
- 在 Spring Boot 上配置 Tomcat 连接池
- 将Camel消息路由到嵌入WildFly的Artemis上
和其它编程语言一样,Go也提供了反射包reflect
供开发者使用。反射机制允许我们在程序运行时检查变量的类型结构、值、方法等,同时还能动态修改变量值、调用方法等。在使用Go的反射操作时,首先需要导入Go的反射包import reflect
对于一个变量来说,最基本的信息就是它的类型和值。在Go的反射包中定义了两个类型reflect.Type
和reflect.Value
来分别表示变量的类型信息和变量的值。同时,定义了下面两个函数:
func TypeOf(i interface{}) Type
func ValueOf(i interface{}) Value
这两个函数分别返回被检查对象的类型和值。它们都是通过一个空接口类型的输入参数传入所要检查的对象。
注意:Value
里面也包含了值的类型信息,可以通过Value.Type()
方法来获取值的类型信息,但Type
里面包含了关于变量类型更多的信息,比如在一些结构体类型,使用Type
就可以获取结构体字段等等更加复杂的操作
此外,reflect.Value
和reflect.Type
都有一个Kind()
方法,它返回一个reflect.Kind
类型的变量,它和reflect.Type
的区别是:Kind
返回的是底层类型,而Type
返回的是静态声明的类型。举个例子,比如我们自定义了类型type zhengxing int
,然后定义了一个变量var i zhengxing
,那么通过它的Type
为zhengxing
,而Kind
为int
下面列举一些通过反射操作来解析一些对象变量的实例:
import (
"fmt"
"reflect"
)
func main(){
var i int = 1
v := reflect.ValueOf(i) //获取Value
fmt.Println(v.Int())
t := reflect.TypeOf(i) //获取Type
fmt.Println(t.Name())
}
上面的代码实现了使用反射机制获取int
类型变量的类型和值,其中v.Int()
是以int
类型返回v
的值,Value
还有其它类似的方法比如Value.Float()
、Value.Bool()
等,如果返回的变量值不是对应的类型,将出现panic
。其次,t.Name()
是返回类型的名称
import (
"fmt"
"reflect"
)
type dog struct {
name string
age int
}
func (d dog) Run(speed string) {
fmt.Println(d.name, "is running", speed)
}
func main(){
d := dog{ "dog", 2}
t := reflect.TypeOf(d)
fmt.Println("type: ", t.Name())
v := reflect.ValueOf(d)
fmt.Println("Fields:")
for i := 0; i < t.NumField(); i++ {
f := t.Field(i)
val := v.Field(i)
fmt.Println(f.Name, f.Type, val)
}
for i := 0; i < t.NumMethod(); i++ {
m := t.Method(i)
fmt.Println(m.Name, m.Type)
}
}
与简单类型相比,结构体中还有多个字段和方法,因此解析起来较为复杂。
在上面的代码中,t := reflect.TypeOf(d)
会得到一个struct
结构体类型,v := reflect.ValueOf(d)
可以得到结构体内字段的值信息。t.NumField()
可以得到结构体内字段的数量,然后根据索引来依次获取字段的类型和值:f := t.Field(i)
、val := v.Field(i)
t.NumMethod()
可以获取t
中的方法个数,通过t.Method(i)
来获取方法
特别需要注意的是,这里结构体的方法需要是首字母大写的(对外部包可见),否则将反射获取不到该方法
type User struct {
Id int
Name string
}
type Manager struct {
User
title string
}
func main(){
m := Manager{
User: User{1, "user"},
title: "kk",
}
t := reflect.TypeOf(m)
fmt.Println(t.Field(0)) //{User main.User 0 [0] true}
fmt.Println(t.Feile(1)) //{title main string 24 [1] false}
fmt.Println(t.FieldByIndex([]int{0, 0})) //{Id int 0 [0] false}
fmt.Println(t.FieldByIndex([]int{0, 1})) //{Name string 8 [1] false}
}
上面代码中注释的部分显示了打印的结果,可以看到通过t.Field(0)
和t.Field(1)
分别得到了字段User
和title
的信息,其中值得注意的是,User
字段信息的最后一个信息是true
,其它的字段为false
,这个布尔值表示该字段是否为匿名字段,从Manager
的定义可以看到User
是一个内嵌结构体,是匿名字段。
t.FieldByIndex()
传入的是一个int
型的slice
,它的第一数字表示外层结构体(Manager
)的字段索引,第二个数字表示内嵌结构体(User
)内字段的索引,从而获取到内嵌结构体中的字段
import "reflect"
func main(){
x := 123
v := reflect.ValueOf(x)
v.Elem().SetInt(999)
}
如上代码所示,通过v.Elem()
获取到v
的值并重新对它进行设置SetInt()
type User struct {
Id int
Name string
}
func main(){
u := User{
Id: 1,
Name: "32",
}
Set(&u)
fmt.Println(u)
}
func Set(o interface{}) {
v := reflect.ValueOf(o)
if v.Kind() != reflect.Ptr{
fmt.Println("can not be set")
return
}
value := v.Elem()
if !value.CanSet(){
fmt.Println("can not be set")
}
f := value.FieldByName("Name")
if !f.IsValid() {
fmt.Println("can not be set")
return
}
if f.Kind() == reflect.String {
f.SetString("wangwang")
}
}
如上代码所示,我们定义了一个Set
函数,在函数中用反射机制来修改结构体中值。在函数中包括了许多类型判断,并且函数传入了一个结构体的地址指针,才能真正地修改结构体中的值
这里同样需要注意,结构体中要修改的字段同样需要是首字母大写的可见类型,否则会报如下异常:
panic: reflect: reflect.flag.mustBeAssignable using value obtained using unexported field
使用反射机制可以在程序中动态调用方法
type dog struct {
Name string
Age int
}
func (d dog) Run(speed string) {
fmt.Println(d.name, "is running", speed)
}
func main(){
d := dog{"gogo", 1}
v := reflect.ValueOf(d)
mv := v.MethodByName("Run")
args := []reflect.Value{reflect.ValueOf("fastly")}
mv.Call(args)
}
通过MethodByName()
方法来返回指定方法名的函数值类型,然后使用它的Call()
方法去动态调用方法,当需要传参时,传入的参数必须时reflect.Value
类型组成的一个切片,可以使用reflect.ValueOf()
方法将任意类型的值转换成一个relfect.Value
类型
关闭。这个问题是opinion-based .它目前不接受答案。 想要改进这个问题? 更新问题,以便 editing this post 可以用事实和引用来回答它. 关闭 9 年前。 Improve
介绍篇 什么是MiniApis? MiniApis的特点和优势 MiniApis的应用场景 环境搭建 系统要求 安装MiniApis 配置开发环境 基础概念 MiniApis架构概述
我正在从“JavaScript 圣经”一书中学习 javascript,但我遇到了一些困难。我试图理解这段代码: function checkIt(evt) { evt = (evt) ? e
package com.fastone.www.javademo.stringintern; /** * * String.intern()是一个Native方法, * 它的作用是:如果字
您会推荐哪些资源来学习 AppleScript。我使用具有 Objective-C 背景的传统 C/C++。 我也在寻找有关如何更好地开发和从脚本编辑器获取更快文档的技巧。示例提示是“查找要编写脚本的
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 4年前关闭。 Improve thi
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 7年前关闭。 Improve thi
关闭。这个问题不符合 Stack Overflow guidelines 。它目前不接受答案。 想改善这个问题吗?更新问题,以便堆栈溢出为 on-topic。 6年前关闭。 Improve this
我是塞内加尔的阿里。我今年60岁(也许这是我真正的问题-笑脸!!!)。 我正在学习Flutter和Dart。今天,我想使用给定数据模型的列表(它的名称是Mortalite,请参见下面的代码)。 我尝试
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题? Update the question所以它是on-topic对于堆栈溢出。 9年前关闭。 Improve this que
学习 Cappuccino 的最佳来源是什么?我从事“传统”网络开发,但我对这个新框架非常感兴趣。请注意,我对 Objective-C 毫无了解。 最佳答案 如上所述,该网站是一个好地方,但还有一些其
我正在学习如何使用 hashMap,有人可以检查我编写的这段代码并告诉我它是否正确吗?这个想法是有一个在公司工作的员工列表,我想从 hashMap 添加和删除员工。 public class Staf
我正在尝试将 jQuery 与 CoffeScript 一起使用。我按照博客中的说明操作,指示使用 $ -> 或 jQuery -> 而不是 .ready() 。我玩了一下代码,但我似乎无法理解我出错
还在学习,还有很多问题,所以这里有一些。我正在进行 javascript -> PHP 转换,并希望确保这些做法是正确的。是$dailyparams->$calories = $calories;一条
我目前正在学习 SQL,以便从我们的 Magento 数据库制作一个简单的 RFM 报告,我目前可以通过导出两个查询并将它们粘贴到 Excel 模板中来完成此操作,我想摆脱 Excel 模板。 我认为
我知道我很可能会因为这个问题而受到抨击,但没有人问,我求助于你。这是否是一个正确的 javascript > php 转换 - 在我开始不良做法之前,我想知道这是否是解决此问题的正确方法。 JavaS
除了 Ruby-Doc 之外,哪些来源最适合获取一些示例和教程,尤其是关于 Ruby 中的 Tk/Tile?我发现自己更正常了 http://www.tutorialspoint.com/ruby/r
我只在第一次收到警告。这正常吗? >>> cv=LassoCV(cv=10).fit(x,y) C:\Python27\lib\site-packages\scikit_learn-0.14.1-py
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
As it currently stands, this question is not a good fit for our Q&A format. We expect answers to be
我是一名优秀的程序员,十分优秀!