一份反射go reflect的API练习以及其坑点

主要内容:

  • 由对象获取反射类型,由对象获取反射值
  • 由反射值获取反射类型
  • 反射值重新转换成对象
  • 遍历字段
  • 遍历方法
  • 易知go语言的struct是值类型,如果修改需要使用引用传递。反射修改值类型,要获取其指针的值类型。此方法适用于int string 以及struct。
  • 反射调用的注意点:
    • 类方法要大写,以便能被其它包,也就是reflect包调用。小写的方法可以由反射查询,但是不能调用。
    • 要获取其引用的反射。
    • 反射调用方法,最好检测m.String是否为<invalid Value>,避免程序因不能正确获取反射方法而发生panic。
    • 即使是调用无形参的反射方法,也要传入[]reflect.Value{}或者nil。
    • 反射调用不定参方法,传入切片打散后,再转成reflect.Value。
  • 最后附上一个反射调用的包装,用来排除以上坑点。
type User struct {
    Id int
    Name string
}

func (c *User)getId() int {
    return c.Id
}

func (c *User)getName() string {
    return c.Name
}

func (c *User)AppendString(s1 string,s2 ...string) (sRes string) {
    sRes = s1
    for _,str := range s2{
        sRes += str
    }
    return
}

func main()  {
    //获取反射信息
    u := User{1,"one"}
    uType := reflect.TypeOf(u)
    uValue := reflect.ValueOf(u)

    uFf := uValue.Interface().(User)
    fmt.Println(uFf)

    uType2 := uValue.Type()
    fmt.Printf("get type from uValue:%s\n",uType2.Name())

    for i:=0;i<uType.NumField();i++{
        fmt.Printf("field:%s,value:%+v\n",uType.Field(i).Name,uValue.Field(i).Interface())
    }

    for i:=0;i<uType.NumMethod();i++{
        m := uType.Method(i)
        fmt.Printf("method name is %s\n",m.Name)
    }

    //对值类型改值
    x := 10
    xF := reflect.ValueOf(x)
    fmt.Printf("could change:%t\n",xF.CanSet())//false值类型不可修改

    xPF := reflect.ValueOf(&x).Elem()
    fmt.Printf("could change value:%t\n",xPF.CanSet())
    xPF.SetInt(11)
    fmt.Printf("changed value is:%v\n",xPF.Interface())

    //反射修改struct
    u1 := User{1,"One"}
    u1PF := reflect.ValueOf(&u1).Elem()
    if u1PF.CanSet(){
        u1PF.Field(0).SetInt(2)
        u1PF.Field(1).SetString("two")
    }
    fmt.Printf("changed struct value is %v\n",u1PF.Interface().(User))

    //反射调用方法
    reflectMethodInput := []reflect.Value{reflect.ValueOf("http"),reflect.ValueOf("Request")}
    m := reflect.ValueOf(&u1).MethodByName("AppendString")
    fmt.Println(m.String())
    mR := m.Call(reflectMethodInput)
    fmt.Println(mR)
}
func reflectCall(obj interface{}, method string, args... interface{}) []reflect.Value{
    argsR := make([]reflect.Value, len(args))
    for i, _ := range args {
        argsR[i] = reflect.ValueOf(args[i])
    }
    if v := reflect.ValueOf(&obj).MethodByName(method); v.String() == "<invalid Value>" {
        return nil
    }else {
        return v.Call(argsR)
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容