• 企业400电话
  • 微网小程序
  • AI电话机器人
  • 电商代运营
  • 全 部 栏 目

    企业400电话 网络优化推广 AI电话机器人 呼叫中心 网站建设 商标✡知产 微网小程序 电商运营 彩铃•短信 增值拓展业务
    go语言通过反射创建结构体、赋值、并调用对应的操作

    我就废话不多说了,大家还是直接看代码吧~

    package main
    import (
    	"fmt"
    	"reflect"
    	"testing"
    )
    type Call struct {
    	Num1 int
    	Num2 int
    }
    func (call Call) GetSub(name string){
    	fmt.Printf("%v 完成了减法运算,%v - %v = %v \n", name, call.Num1, call.Num2, call.Num1 - call.Num2)
    }
    func (call *Call) GetSum(name string){
    	fmt.Printf("%v 完成了加法运算,%v + %v = %v \n", name, call.Num1, call.Num2, call.Num1 + call.Num2)
    }
    func TestReflect(t *testing.T) {
    	var (
    		call *Call
    		rValues []reflect.Value
    		rValues2 []reflect.Value
    	)
    	ptrType := reflect.TypeOf(call) //获取call的指针的reflect.Type
    	trueType := ptrType.Elem() //获取type的真实类型
    	ptrValue := reflect.New(trueType) //返回对象的指针对应的reflect.Value
    	call = ptrValue.Interface().(*Call)
    	trueValue := ptrValue.Elem() //获取真实的结构体类型
    	trueValue.FieldByName("Num1").SetInt(123)//设置对象属性,注意这个一定要是真实的结构类型的reflect.Value才能调用,指针类型reflect.Value的会报错
    	//ptrValue.FieldByName("Num2").SetInt(23)
    	trueValue.FieldByName("Num2").SetInt(23)
    	//rValues = make([]reflect.Value, 0)
    	rValues = append(rValues, reflect.ValueOf("xiaopeng"))//调用对应的方法
    	fmt.Println(rValues)
    	trueValue.MethodByName("GetSub").Call(rValues)
    	/*
    	fixme 在反射中,指针的方法不可以给实际类型调用,实际类型的方法可以给指针类型调用,因为go语言对这种操作做了封装
    	所以下面一句是没问题的
    	下下一句会运行时报错
    	 */
    	//ptrValue.MethodByName("GetSub").Call(rValues)
    	//trueValue.MethodByName("GetSum").Call(append(rValues2, reflect.ValueOf("hiram")))
    	ptrValue.MethodByName("GetSum").Call(append(rValues2, reflect.ValueOf("hiram")))
    	fmt.Println(call)
    	
    	/*
    	fixme 在实际使用中  指针和实体都能相互转换,不会影响调用
    	但是指针的方法在方法体内的操作会影响到结构体本身属性
    	而实体的方法不会,因为go对于结构体、数组、基本类型都是值传递
    	 */
    	call.GetSub("aaa")
    	(*call).GetSub("bbb")
    	call.GetSum("ccc")
    	(*call).GetSum("ddd")
    }

    补充:golang 反射 reflect 设置 struct 字段

    说明1 reflect.Value区分CanSet和Can not Set

    所以, 必须要返回成Can set的reflect.Value

    如:

    s := reflect.ValueOf(t).Elem()

    然后就可以happy的设值了, 可是不能随便设值的, 一个通用的方法就是使用Set(v Value)方法,

    说明2 将值转成reflect.Value类型

    下面的这段代码就是转成Value类型

    sliceValue := reflect.ValueOf([]int{1, 2, 3}) // 这里将slice转成reflect.Value类型

    说明3 reflect.ValueOf 参数必须是一个 指针 或 interface Elem()才可以正常调用

    func (Value) Elem func (v Value) Elem() Value

    Elem returns the value that the interface v contains or that the pointer v points to. It panics if v's Kind is not Interface or Ptr. It returns the zero Value if v is nil.

    Elem返回接口v包含的值或指针v指向的值。 如果v的Kind不是Interface或Ptr,它会感到恐慌。 如果v为零,它将返回零值。

    实例代码

    代码1:

    func Destroy(subj interface{}) {
    	stype := reflect.ValueOf(subj).Elem()
    	field := stype.FieldByName("Status")
    	if field.IsValid() {
    		field.SetString("Destroyed")
    	}
    } 
     
    func TestDestroy(t *testing.T) {
    	// Initialize data
    	jaeger := Jaeger{Name: "Cherno Alpha", Country: "RU", Status: "Active"}
    	kaiju := Kaiju{Alias: "Scissure", Origin: "Sydney", Status: "Unknown"}
    	shatterdome := Shatterdome{Location: "Lima"}
     
    	// Destroy everything
    	Destroy(jaeger)
    	Destroy(kaiju)
    	Destroy(shatterdome)
     
    	// Check the result
    	if jaeger.Status != "Destroy" {
    		t.Error("jaeger was not destroyed")
    	}
    	if kaiju.Status != "Destroy" {
    		t.Error("kaiju was not destroyed")
    	}
    }

    代码2:

    type T struct {
        Age int
        Name string
        Children []int
    }
    t := T{12, "someone-life", nil}
    s := reflect.ValueOf(t).Elem()
     
    s.Field(0).SetInt(123) // 内置常用类型的设值方法
    sliceValue := reflect.ValueOf([]int{1, 2, 3}) // 这里将slice转成reflect.Value类型
    s.FieldByName("Children").Set(sliceValue)
     

    以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。如有错误或未考虑完全的地方,望不吝赐教。

    您可能感兴趣的文章:
    • Go语言之结构体与方法
    • Go语言基础语法之结构体及方法详解
    • Go语言-为什么返回值为接口类型,却返回结构体
    • go语言使用第三方包 json化结构体操作示例
    • 关于Go 空结构体的 3 种使用场景
    上一篇:go中控制goroutine数量的方法
    下一篇:golang coroutine 的等待与死锁用法
  • 相关文章
  • 

    © 2016-2020 巨人网络通讯 版权所有

    《增值电信业务经营许可证》 苏ICP备15040257号-8

    go语言通过反射创建结构体、赋值、并调用对应的操作 语言,通过,反射,创建,结构,