Go 反射案例实践

反射案例实践

  • 使用反射来遍历结构体的字段调用结构体的方法,并获取结构体标的值

案例实践所需要的参考知识


func (Value) NumMethod

func (v Value) NumMethod() int
  • 返回v持有值的方法集的方法数目

func (Value) Method

func (v Value) Method(i int) Value
  • 返回v持有值类型第i个方法(默认从0开始)的已绑定(到v的持有值的)状态的函数形式的Value封装。
  • 返回值调用Call方法时不应包含接收者
  • 返回值持有的函数总是使用v的持有者作为接收者(即第一个参数)。
  • 如果i出界,或者v的持有值是接口类型的零值(nil),会panic。

func (Value) Call

func (v Value) Call(in []Value) []Value
  • Call方法使用输入的参数in调用v持有的函数
    • 例如,如果len(in) == 3,v.Call(in)代表调用v(in[0], in[1], in[2])(其中Value值表示其持有值)。
  • 如果v的Kind不是Func会panic。
  • 它返回函数所有输出结果的Value封装的切片。
  • 和go代码一样,每一个输入实参的持有值都必须可以直接赋值给函数对应输入参数的类型。
  • 如果v持有值是可变参数函数,Call方法会自行创建一个代表可变参数的切片,将对应可变参数的值都拷贝到里面。

func (Value) NumField

func (v Value) NumField() int
  • 返回v持有的结构体类型值的字段数,如果v的Kind不是Struct会panic

func (Value) Field

func (v Value) Field(i int) Value
  • 返回结构体的第i个字段(的Value封装)。如果v的Kind不是Struct或i出界会panic

type Type

type Type interface {
    //...

     Field(i int) StructField
    返回索引序列指定的嵌套字段的类型,
    等价于用索引中每个值链式调用本方法,如非结构体将会panic
    //...
}

type StructField

type StructField struct {
    // Name是字段的名字。PkgPath是非导出字段的包路径,对导出字段该字段为""。
    // 参见http://golang.org/ref/spec#Uniqueness_of_identifiers
    Name    string
    PkgPath string
    Type      Type      // 字段的类型
    Tag       StructTag // 字段的标签
    Offset    uintptr   // 字段在结构体中的字节偏移量
    Index     []int     // 用于Type.FieldByIndex时的索引切片
    Anonymous bool      // 是否匿名字段
}
  • StructField类型描述结构体中的一个字段的信息

type StructTag

type StructTag string
  • StructTag是结构体字段的标签。
  • 一般来说,标签字符串是(可选的)空格分隔的一连串   `key:"value"`  对
  • 每个键都是不包含控制字符、空格、双引号、冒号的非空字符串。每个值都应被双引号括起来,使用go字符串字面语法。

func (StructTag) Get

func (tag StructTag) Get(key string) string
  • Get方法返回标签字符串中键key对应的值
  • 如果标签中没有该键,会返回""。
  • 如果标签不符合标准格式,Get的返回值是不确定的。

案例实践

package main
import (
	"fmt"
	"reflect"
)
//定义了一个Monster结构体
type Monster struct {
	Name  string `json:"name"`
	Age   int `json:"monster_age"`
	Score float32 `json:"成绩"`
	Sex   string
	
}

//方法,返回两个数的和
func (s Monster) GetSum(n1, n2 int) int {
	return n1 + n2
}
//方法, 接收四个值,给s赋值
func (s Monster) Set(name string, age int, score float32, sex string) {
	s.Name = name
	s.Age = age
	s.Score = score
	s.Sex = sex
}

//方法,显示s的值
func (s Monster) Print() {
	fmt.Println("---start~----")
	fmt.Println(s)
	fmt.Println("---end~----")
}
func TestStruct(a interface{}) {
	//获取reflect.Type 类型
	typ := reflect.TypeOf(a)
	//获取reflect.Value 类型
	val := reflect.ValueOf(a)
	//获取到a对应的类别
	kd := val.Kind()
	//如果传入的不是struct,就退出
	if kd !=  reflect.Struct {
		fmt.Println("expect struct")
		return
	}

	//获取到该结构体有几个字段
	num := val.NumField()

	fmt.Printf("struct has %d fields\n", num) //4
	//变量结构体的所有字段
	for i := 0; i < num; i++ {
		fmt.Printf("Field %d: 值为=%v\n", i, val.Field(i))
		//获取到struct标签, 注意需要通过reflect.Type来获取tag标签的值
		tagVal := typ.Field(i).Tag.Get("json")
		//如果该字段于tag标签就显示,否则就不显示
		if tagVal != "" {
			fmt.Printf("Field %d: tag为=%v\n", i, tagVal)
		}
	}
	
	//获取到该结构体有多少个方法
	numOfMethod := val.NumMethod()
	fmt.Printf("struct has %d methods\n", numOfMethod)
	
	//var params []reflect.Value
	//方法的排序默认是按照 函数名的排序(ASCII码)
	val.Method(1).Call(nil) //获取到第二个方法。调用它

	
	//调用结构体的第1个方法Method(0)
	var params []reflect.Value  //声明了 []reflect.Value
	params = append(params, reflect.ValueOf(10))
	params = append(params, reflect.ValueOf(40))
	res := val.Method(0).Call(params) //传入的参数是 []reflect.Value, 返回[]reflect.Value
	fmt.Println("res=", res[0].Int()) //返回结果, 返回的结果是 []reflect.Value*/

}
func main() {
	//创建了一个Monster实例
	var a Monster = Monster{
		Name:  "黄鼠狼精",
		Age:   400,
		Score: 30.8,
	}
	//将Monster实例传递给TestStruct函数
	TestStruct(a)	
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值