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")
}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM