Go基础之--反射

时间:2022-05-06
本文章向大家介绍Go基础之--反射,主要内容包括反射中重要函数的演示、获取变量的值、通过反射来改变变量的值、反射操作结构体、基本概念、基础应用、原理机制和需要注意的事项等,并结合实例形式分析了其使用技巧,希望通过本文能帮助到大家理解应用这部分内容。

反射:可以在运行时动态获取变量的相关信息

反射需要导入reflect

反射中重要函数的演示

反射有几下几个重要的函数: reflect.TypeOf :获取变量的类型,返回reflect.Type类型 reflect.ValueOf:获取变量的值,返回reflect.Value类型 reflect.Value.Kind:获取变量的类别,返回一个常量 reflect.Value.Interface():转换成interface{}类型

通过一个小例子来理解:

package main

import (
    "reflect"
    "fmt"
)

type Student struct{
    Name string
    Age int
}

func (s*Student) SetName(name string){
    s.Name="coders"
}

func (s*Student)SetAge(age int){
    s.Age = 23
}


func getTypeInfo(a interface{}){
    // 用于获取一个数据的数据类型
    typeInfo := reflect.TypeOf(a)
    kind := typeInfo.Kind()
    fmt.Println("kind of a :",kind)

    num := typeInfo.NumMethod() //获取当前数据有多少个方法
    fmt.Println("method num:",num)

    method,ok:=typeInfo.MethodByName("SetName") //获取是否有某个方法
    if !ok{
        fmt.Println("not have method SetName")
    }else{
        fmt.Println(method)
    }
}

func getAllMethod(a interface{}){
    // 用于获取变量下的所有方法
    typeInfo := reflect.TypeOf(a)
    num := typeInfo.NumMethod()
    for i:=0;i<num;i++ {
        method:= typeInfo.Method(i)
        fmt.Println(method)
    }
}

func testGetAllMethod()  {
    var stu Student
    getAllMethod(&stu)
}

func testGetTypeInfo(){
    var i int
    getTypeInfo(i) //获取的结果就是int

    var stu Student
    getTypeInfo(&stu) //获取的结果就是struct
    getAllMethod(&stu)

    var s []int
    getTypeInfo(s)  //获取的结果就是slice

    var a [5]int
    getTypeInfo(a)  //获取的结果就是array


}

func testGetValueInfo(){
    var i = 100
    valueInfo := reflect.ValueOf(i)
    tmp := valueInfo.Interface()  //转换成interface类型
    val := tmp.(int) //这里我是知道是int所以直接转换了
    fmt.Println("val:",val) //这里获取的还是100
    fmt.Println("val of valueInfo:",valueInfo.Int()) // 这里打印的也是100
    fmt.Println("type:",valueInfo.Type())
    fmt.Println("kind:",valueInfo.Kind())
}

func main(){
    testGetTypeInfo()
    testGetAllMethod()
    testGetValueInfo()
}

上面这个例子中演示了reflect.Value.Kind()可以返回int,struct,slice,array,当然这里可以返回的类型还有很多如下: Bool Int Int8 Int16 Int32 Int64 Uint Uint8 Uint16 Uint32 Uint64 Uintptr Float32 Float64 Complex64 Complex128 Array Chan Func Interface Map Ptr Slice String Struct UnsafePointer

获取变量的值

reflect.ValueOf(x).Float() reflect.ValueOf(x).Int() reflect.ValueOf(x).String() reflect.ValueOf(x).Bool()

这个功能在上面的代码中也有演示

通过反射来改变变量的值

reflect.Value.SetXX相关方法,如: reflect.Value.SetFloat():设置浮点数 reflect.Value.SetInt():设置整数 reflect.Value.SetString():设置字符串

通过下面一个简单的例子来演示:

package main

import (
    "reflect"
    "fmt"
)

func main() {
    var a float64
    fmt.Println(a)
    fv := reflect.ValueOf(a)
    fv.SetFloat(3.14)
    fmt.Println(a)
}

上面这段代码会提示如下错误:

这里需要知道的是我们的变量a是一个值类型的变量,我们通过reflect.valueOf传入的时候其实是传入的变量的拷贝,所以我们如果通过SetFloat给变量设置值的时候其实并不会生效,go这里已经替我考虑到了,所以给我们提示了上面这个错误信息,那是不是我们在reflect.Value的传入地址就可以了呢,我把上述代码中更改为:reflect.Value(&a),当我们运行后发现还是报了和上面相同的错误,这是为什么呢?

我们应该还记得如果是一个指针的时候我们赋值的时候是需要在指针的左边写个*符号,但是这是在反射里面我们怎么写星号,所以go在这里提供给我们另外一个方法,当我们通过调用SetFloat的时候用: fv.Elem().SetFloat(3.14)这种方式调用就ok了,完整的正确代码为:

package main

import (
    "reflect"
    "fmt"
)

func main() {
    var a float64
    fmt.Println(a)
    fv := reflect.ValueOf(&a)
    fv.Elem().SetFloat(3.14)
    fmt.Println(a)
}

反射操作结构体

reflect.Value.NumField():获取结构体中字段的个数 reflect.Value.Method(n).Call():调用结构体中的方法

package main

import (
    "reflect"
    "fmt"
)

type Student struct{
    Name string
    Age int
    Sex int
}

func (s *Student) Set(name string,age int,sex int){
    s.Name = name
    s.Age = age
    s.Sex = sex
}

func testStruct()  {
    var stu *Student = &Student{}
    stu.Set("coder",23,1)
    valueInfo := reflect.ValueOf(stu)

    fieldNum := valueInfo.Elem().NumField()
    fmt.Println("filed num:",fieldNum) //这里返回的结果是3

    sexValueInfo := valueInfo.Elem().FieldByName("Sex")
    fmt.Println("sex=",sexValueInfo.Int())
    sexValueInfo.SetInt(0) //这里是更改值
    fmt.Println(stu)
    setMethod := valueInfo.MethodByName("Set") //获取Set方法
    var params []reflect.Value
    name := "tom"
    age := 18
    sex:=2
    params = append(params,reflect.ValueOf(name),reflect.ValueOf(age),reflect.ValueOf(sex))
    setMethod.Call(params) //调用Set方法
    fmt.Println(stu) //将最开始的值已经更改了


}

func main() {
    testStruct()
}