反射的坑

主要内容:

  • 由对象获取反射类型,由对象获取反射值
  • 由反射值获取反射类型
  • 反射值重新转换成对象
  • 遍历字段
  • 遍历方法
  • 易知go语言的struct是值类型,如果修改需要使用引用传递。反射修改值类型,要获取其指针的值类型。此方法适用于int string 以及struct。
  • 反射调用的注意点:
    • 类方法要大写,以便能被其它包,也就是reflect包调用。小写的方法可以由反射查询,但是不能调用。
    • 要获取其引用的反射。
    • 反射调用方法,最好检测m.String是否为,避免程序因不能正确获取反射方法而发生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必须是指针类型,而且值本身可以修改,或者是结构体的某个大写字段才会返回true,否则都是false,比如下面
    /*type student struct {
        name string
        Age  int
    }
      ss := reflect.ValueOf(&stu2).Elem()
      //canSet返回真条件1必须是指针类型,Field拿到的字段必须是可以导出即大写,
      fmt.Println(ss.Field(0).CanSet()) 这里为false,因为拿到的是name是小写
      fmt.Println(ss.Field(1).CanSet()) 这里是true,因为拿到的是Age是大写
      */
    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)
    }
}




posted @ 2019-12-19 19:35  离地最远的星  阅读(217)  评论(0编辑  收藏  举报