Go 反射机制详解
反射的本质就是在程序运行的时候,获取对象的类型信息和内存结构,反射是把双刃剑,功能强大但可读性差,反射代码无法在编译阶段静态发现错误,反射的代码常常比正常代码效率低1~2个数量级,如果在关键位置使用反射会直接导致代码效率问题,所以,如非必要,不建议使用。
静态类型是指在编译的时候就能确定的类型(常见的变量声明类型都是静态类型);动态类型是指在运行的时候才能确定的类型(比如接口,也只有接口才有反射)。
使用反射的三个步骤:
- 先有一个接口类型的变量
- 把它转成reflect对象 一般就是type 或者 value类型
- 然后根据不同的情况调用相应的函数
TypeOf() ValueOf()
为了说明其用法,先举个最简单的例子:
package main
import (
"fmt"
"reflect"
)
func main() {
var x float64 = 3.4
fmt.Println("type : ", reflect.TypeOf(x))
fmt.Println("value : ", reflect.ValueOf(x))
}
运行结果是:
type : float64
value : 3.4
获取接口变量信息
事先知道原有类型的时候
举个例子:
package main
import (
"fmt"
"reflect"
)
func main() {
var num float64 = 3.14
//接口类型变量得到一个反射类型的变量
value := reflect.ValueOf(num)
//从一个反射类型对象得到接口类型变量
conervtValue := value.Interface().(float64)
fmt.Println(conervtValue)
//pointer 包含了一个float64的指针类型
pointer := reflect.ValueOf(&num)
convertPointer := pointer.Interface().(*float64)
fmt.Println(convertPointer)
}
运行结果是:
3.14
0x1400012a008
事先不知道原有类型的时候
这时候我们一般需要遍历探测一下Field
举个例子:
package main
import (
"fmt"
"reflect"
)
type Person struct {
Name string
Age int
Gender string
}
func (p Person) Say(msg string) {
fmt.Println("hello, ", msg)
}
func (p Person) PrintInfo() {
fmt.Printf("Name: %s, Age: %d, Gender: %s", p.Name, p.Age, p.Gender)
}
func main() {
p1 := Person{"bill", 16, "Male"}
GetMessage(p1)
}
//获取input的信息 在这个函数中 输入是空接口
//代表我们并不知道input的原始类型是什么 取决于函数调用的时候掺进来什么参数
func GetMessage(input interface{}) {
getType := reflect.TypeOf(input)
fmt.Println("输入数据的类型是: ", getType.Name())
fmt.Println("输入数据的种类是: ", getType.Kind())
getValue := reflect.ValueOf(input)
fmt.Println("all fields are: ", getValue)
}
运行结果如下:
输入数据的类型是: Person
输入数据的种类是: struct
all fields are: {bill 16 Male}
上面的例子,我们一口气把所有的字段值全部打印出来了,但如果我们想挨个打印每个字段的名字,类型,数值我们应该这样做:
package main
import (
"fmt"
"reflect"
)
type Person struct {
Name string
Age int
Gender string
}
func (p Person) Say(msg string) {
fmt.Println("hello, ", msg)
}
func (p Person) PrintInfo() {
fmt.Printf("Name: %s, Age: %d, Gender: %s", p.Name, p.Age, p.Gender)
}
func main() {
p1 := Person{"bill", 16, "Male"}
GetMessage(p1)
}
//获取input的信息 在这个函数中 输入是空接口
//代表我们并不知道input的原始类型是什么 取决于函数调用的时候掺进来什么参数
func GetMessage(input interface{}) {
getType := reflect.TypeOf(input)
fmt.Println("输入数据的类型是: ", getType.Name())
fmt.Println("输入数据的种类是: ", getType.Kind())
getValue := reflect.ValueOf(input)
fmt.Println("all fields are: ", getValue)
//获取字段
for i := 0; i < getType.NumField(); i++ {
field := getType.Field(i)
value := getValue.Field(i).Interface()
fmt.Printf("字段名称: %s, 字段类型: %s, 字段值: %v\n ", field.Name, field.Type, value)
}
}
运行结果是:
输入数据的类型是: Person
输入数据的种类是: struct
all fields are: {bill 16 Male}
字段名称: Name, 字段类型: string, 字段值: bill
字段名称: Age, 字段类型: int, 字段值: 16
字段名称: Gender, 字段类型: string, 字段值: Male
如果我们还想获取方法怎么办呢?原理和上面差不多,不过我们需要把field改成method,举个例子:
package main
import (
"fmt"
"reflect"
)
type Person struct {
Name string
Age int
Gender string
}
func (p Person) Say(msg string) {
fmt.Println("hello, ", msg)
}
func (p Person) PrintInfo() {
fmt.Printf("Name: %s, Age: %d, Gender: %s", p.Name, p.Age, p.Gender)
}
func main() {
p1 := Person{"bill", 16, "Male"}
GetMessage(p1)
}
//获取input的信息 在这个函数中 输入是空接口
//代表我们并不知道input的原始类型是什么 取决于函数调用的时候掺进来什么参数
func GetMessage(input interface{}) {
getType := reflect.TypeOf(input)
fmt.Println("输入数据的类型是: ", getType.Name())
fmt.Println("输入数据的种类是: ", getType.Kind())
getValue := reflect.ValueOf(input)
fmt.Println("all fields are: ", getValue)
//获取字段
for i := 0; i < getType.NumField(); i++ {
field := getType.Field(i)
value := getValue.Field(i).Interface()
fmt.Printf("字段名称: %s, 字段类型: %s, 字段值: %v\n ", field.Name, field.Type, value)
}
//获取方法
for i := 0; i < getType.NumMethod(); i++ {
method := getType.Method(i)
fmt.Printf("方法名称: %s, 方法类型: %v\n", method.Name, method.Type)
}
}
运行结果是:
输入数据的类型是: Person
输入数据的种类是: struct
all fields are: {bill 16 Male}
字段名称: Name, 字段类型: string, 字段值: bill
字段名称: Age, 字段类型: int, 字段值: 16
字段名称: Gender, 字段类型: string, 字段值: Male
方法名称: PrintInfo, 方法类型: func(main.Person)
方法名称: Say, 方法类型: func(main.Person, string)
参考:bilibili
推荐阅读
CSS in JS 之 Styled-components 用法
npm i styled-components基本用法 {代码...} 支持sass/less语法 {代码...} 可透传props 以及 基于props做样式判断 {代码...} 样式化任意组件 {代码...} css的复用、扩展、增强 {代码...} 添加动画 定义keyframes ...
LiberHome赞 1阅读 115
Golang 中 []byte 与 string 转换
string 类型和 []byte 类型是我们编程时最常使用到的数据结构。本文将探讨两者之间的转换方式,通过分析它们之间的内在联系来拨开迷雾。
机器铃砍菜刀赞 24阅读 58.4k评论 2
万字详解,吃透 MongoDB!
MongoDB 是一个基于 分布式文件存储 的开源 NoSQL 数据库系统,由 C++ 编写的。MongoDB 提供了 面向文档 的存储方式,操作起来比较简单和容易,支持“无模式”的数据建模,可以存储比较复杂的数据类型,是一款非常...
JavaGuide赞 8阅读 1.7k
数据结构与算法:二分查找
一、常见数据结构简单数据结构(必须理解和掌握)有序数据结构:栈、队列、链表。有序数据结构省空间(储存空间小)无序数据结构:集合、字典、散列表,无序数据结构省时间(读取时间快)复杂数据结构树、 堆图二...
白鲸鱼赞 9阅读 5.4k
PHP转Go实践:xjson解析神器「开源工具集」
我和劲仔都是PHP转Go,身边越来越多做PHP的朋友也逐渐在用Go进行重构,重构过程中,会发现php的json解析操作(系列化与反序列化)是真的香,弱类型语言的各种隐式类型转换,很大程度的减低了程序的复杂度。
王中阳Go赞 11阅读 2.7k评论 4
Git操作不规范,战友提刀来相见!
年终奖都没了,还要扣我绩效,门都没有,哈哈。这波骚Git操作我也是第一次用,担心闪了腰,所以不仅做了备份,也做了笔记,分享给大家。问题描述小A和我在同时开发一个功能模块,他在优化之前的代码逻辑,我在开...
王中阳Go赞 6阅读 2.9k评论 4
妙啊,空结构体还能这么用?Go语言的结构体看这篇就够了
本文详解了Go语言结构体的各个知识点,最后介绍了空结构体的3种妙用。希望对你有帮助。定义结构体,是一种自定义的数据类型,由多个数据类型组合而成。用于描述一类事物相关属性。定义方式: {代码...} 实例化结...
王中阳Go赞 6阅读 1.2k
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。