当前位置: 首页 > news >正文

Go反射四讲---第三讲:如何使用反射操作函数,获取函数的相关信息?

反射-函数

这是我们反射四讲的第三讲,本次给大家讲解如何使用反射处理函数相关的操作。

在这一部分,向大家展示如何输出方法的信息并执行调用。

输出信息,包含方法名,方法参数,返回值。

最后,如何使用反射调用函数。

提醒:在实现时候,我们要非常注意指针的使用,不然会出现 Bug。

测试函数

type User struct {Name stringAge  int
}func NewUserPoint(name string, age int) *User {return &User{Name: name,Age:  age,}
}func NewUser(name string, age int) User {return User{Name: name,Age:  age,}
}func (u User) GetAge() int {return u.Age
}func (u *User) ChangeName(NewName string) {u.Name = NewName
}func (u *User) private() {fmt.Println("private")
}

定义元数据

type FuncInfo struct {Name   string         // 方法名In     []reflect.Type // 方法输入参数Out    []reflect.Type // 方法输出参数Result []any          
}

使用调用

// CallMethod 输出方法的信息并执行调用, 输出:方法名,方法参数,返回值
func CallMethod(val any) (map[string]*FuncInfo, error) {typ := reflect.TypeOf(val)if typ.Kind() != reflect.Struct && typ.Kind() != reflect.Ptr {return nil, errors.New("非法类型")}// 构建结果集num := typ.NumMethod()result := make(map[string]*FuncInfo, num)for i := 0; i < num; i++ {fc := typ.Method(i)// 输入numIn := fc.Type.NumIn()ps := make([]reflect.Value, 0, fc.Type.NumIn())// 第一个参数永远是接收者,ps = append(ps, reflect.ValueOf(val))in := make([]reflect.Type, 0, fc.Type.NumIn())for j := 0; j < numIn; j++ {p := fc.Type.In(j)in = append(in, p)if j > 0 {ps = append(ps, reflect.Zero(p))}}// 调用方法ret := fc.Func.Call(ps)// 输出outNum := fc.Type.NumOut()out := make([]reflect.Type, 0, outNum)res := make([]any, 0, outNum)for k := 0; k < outNum; k++ {out = append(out, fc.Type.Out(k))res = append(res, ret[k].Interface())}result[fc.Name] = &FuncInfo{Name:   fc.Name,In:     in,Out:    out,Result: res,}}return result, nil
}

测试:

func TestCallMethod(t *testing.T) {testcases := []struct {name    stringinput   anywantRes map[string]*FuncInfowantErr error}{{name:  "normal struct",input: types.User{},wantRes: map[string]*FuncInfo{"GetAge": {Name:   "GetAge",In:     []reflect.Type{reflect.TypeOf(types.User{})},Out:    []reflect.Type{reflect.TypeOf(0)},Result: []any{0},},},},{// 指针name:  "pointer",input: &types.User{},wantRes: map[string]*FuncInfo{"GetAge": {Name:   "GetAge",In:     []reflect.Type{reflect.TypeOf(&types.User{})},Out:    []reflect.Type{reflect.TypeOf(0)},Result: []any{0},},"ChangeName": {Name:   "ChangeName",In:     []reflect.Type{reflect.TypeOf(&types.User{}), reflect.TypeOf("")},Out:    []reflect.Type{},Result: []any{},},},},}for _, tt := range testcases {t.Run(tt.name, func(t *testing.T) {res, err := CallMethod(tt.input)assert.Equal(t, tt.wantErr, err)if err != nil {return}assert.Equal(t, tt.wantRes, res)})}
}

总结

对于方法接收器:

  • 以结构体作为输入,那么只能访问到结构体作为接收器的方法
  • 以指针作为输入,那么能访问到任何接收器的方法

输入的第一个参数,永远都是接收器本身。


http://www.mrgr.cn/news/15315.html

相关文章:

  • VCTP(Visual Chain-of-Thought Prompting for Knowledge-Based Visual Reasoning)论文
  • 智能优化特征选择|基于鲸鱼WOA优化算法实现的特征选择研究Matlab程序(XGBoost分类器)
  • docker 安装的mysql8 设置sql_mode
  • 【数据结构-二维前缀和】力扣1314. 矩阵区域和
  • 【Java 设计模式】Callback 模式:掌握异步通信
  • Docker构建镜像时本地NuGet不存在的解决方式
  • 开源 AI 智能名片 O2O 商城小程序在社交私域中的圈层价值
  • 生成和应用patch
  • Datawhale AI夏令营第五期【深度学习进阶】深度学习基础
  • 数学基础 -- 线性代数之行阶梯形
  • netty编程之实现HTTP服务
  • 【Git】修复Git RPC Failed HTTP 500 curl 22错误
  • 抖音ip会莫名其妙变成北京吗
  • 【数据结构】-----哈希
  • Layer Normalization(层归一化)里的可学习的参数
  • ts 新版的@nestjs/commo下redis的注册使用
  • 【Pyspark-驯化】一文搞懂Pyspark中表连接的使用技巧
  • Java Email发送:如何配置SMTP服务器发信?
  • 亚马逊aws的弹性与可扩展性解析
  • 选择排序【详解】