Go 反射机制详解

周阳
• 阅读 1793

    反射的本质就是在程序运行的时候,获取对象的类型信息和内存结构,反射是把双刃剑,功能强大但可读性差,反射代码无法在编译阶段静态发现错误,反射的代码常常比正常代码效率低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

点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
3年前
java反射大全
作者对反射的理解:方法的调用(正常的调用:对象.方法()。反射调用方法:方法.对象())静态属性的调用(正常的调用:类.属性。反射调用:属性.类)常见反射的用法:        1.通过反射获取类Class<?demo1Class
Wesley13 Wesley13
3年前
java中 什么是反射?
JAVA反射机制是在运行状态中,对于任意一个实体类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。JAVA反射(放射)机制:“程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言(https://www.oschina.net/act
Wesley13 Wesley13
3年前
java面试(反射)05
1.什么是反射JAVA反射机制是在运行状态中,对于任意一个类,都能够获取这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取类信息以及动态调用对象内容就称为java语言的反射机制。2.反射的作用在运行时判断任意一个对象所属的
二面京东被问到Java 反射,我直呼好家伙,这我不是必过吗?
二面京东被问到Java反射,我直呼好家伙,这我不是必过吗?用多久我会升职加薪、当上技术总监、迎娶漂亮学姐、走上人生巅峰!想想还有点小激动。好了开始分享面试经历说说你反射的理解到底什么是反射呢???反射的核心就是JVM在运行时才动态加载类或调用方法,访问属性,它不需要事先(写代码的时候或编译期)知道运行对象是谁。每一个类都会产生一个对应的Class对象,
Wesley13 Wesley13
3年前
ulua 动态反射在IOS上的问题
ulua同时支持静态代码生成和动态接口反射。以前在cocos2d中,c没有反射能力,自然需要将所有的接口都静态生成代码给lua使用。c动态反射给lua使用,在ios的il2cpp中存在一个,如果一个类,例如unityengine.ui.inputfield.submitevent这个类,继承自unityevent<string,
Wesley13 Wesley13
3年前
C#进阶之路(七)反射的应用
反射在C中的应用还是很多的,但它对代码的性能有一定影响。反射的性能:  使用反射来调用类型或者触发方法,或者访问一个字段或者属性时clr需要做更多的工作:校验参数,检查权限等等,所以速度是非常慢的。所以尽量不要使用反射进行编程,对于打算编写一个动态构造类型(晚绑定)的应用程序,可以采取以下的几种方式进行代替:1、通过类的继承关系。让该
Wesley13 Wesley13
3年前
Java反射技术概述
1.什么是Java反射?  就是正在运行,动态获取这个类的所有信息2.反射机制的作用  a.反编译:.class.java  b.通过反射机制,访问Java对象的属性,方法,构造方法等3.反射机制的应用场景  Jdbc加载驱动  SpringIOC实现  Java框架4.创建对象的两种方式  a.直
Wesley13 Wesley13
3年前
Java反射的使用姿势一览
反射的学习使用日常的学习工作中,可能用到反射的地方不太多,但看看一些优秀框架的源码,会发现基本上都离不开反射的使用;因此本篇博文将专注下如何使用本片博文布局如下:1.反射是什么,有什么用,可以做什么2.如何使用反射3.实例:利用反射方式,获取一个类的所有成员变量的
Wesley13 Wesley13
3年前
Java重点基础:反射机制
一、什么是反射?Java反射说的是在运行状态中,对于任何一个类,我们都能够知道这个类有哪些方法和属性。对于任何一个对象,我们都能够对它的方法和属性进行调用。我们把这种动态获取对象信息和调用对象方法的功能称之为反射机制。二、反射的三种方式
Wesley13 Wesley13
3年前
Java反射机制及适用场景
什么是Java反射机制?JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的以及动态调用对象的方法的功能称为Java的反射机制。反射的适用场景是什么?1.当你做一个软件可以安装插件的功能,你连插件的类型名称都不知道,你怎么实例化这个对象呢
Java反射源码学习之旅 | 京东云技术团队
在我刚开始了解反射这个Java特性的时候,几乎看到的每一篇文章都会有“Java反射不能频繁使用”、“反射影响性能”之类的话语,当时只是当一个结论记下了这些话,却没有深究过为什么,所以正好借此机会来探究一下Java反射的代码。