go語言通過反射創(chuàng)建結(jié)構(gòu)體、賦值、并調(diào)用對應(yīng)的操作
我就廢話不多說了,大家還是直接看代碼吧~
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) //返回對象的指針對應(yīng)的reflect.Value call = ptrValue.Interface().(*Call) trueValue := ptrValue.Elem() //獲取真實的結(jié)構(gòu)體類型 trueValue.FieldByName("Num1").SetInt(123)//設(shè)置對象屬性,注意這個一定要是真實的結(jié)構(gòu)類型的reflect.Value才能調(diào)用,指針類型reflect.Value的會報錯 //ptrValue.FieldByName("Num2").SetInt(23) trueValue.FieldByName("Num2").SetInt(23) //rValues = make([]reflect.Value, 0) rValues = append(rValues, reflect.ValueOf("xiaopeng"))//調(diào)用對應(yīng)的方法 fmt.Println(rValues) trueValue.MethodByName("GetSub").Call(rValues) /* fixme 在反射中,指針的方法不可以給實際類型調(diào)用,實際類型的方法可以給指針類型調(diào)用,因為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 在實際使用中 指針和實體都能相互轉(zhuǎn)換,不會影響調(diào)用 但是指針的方法在方法體內(nèi)的操作會影響到結(jié)構(gòu)體本身屬性 而實體的方法不會,因為go對于結(jié)構(gòu)體、數(shù)組、基本類型都是值傳遞 */ call.GetSub("aaa") (*call).GetSub("bbb") call.GetSum("ccc") (*call).GetSum("ddd") }
補(bǔ)充:golang 反射 reflect 設(shè)置 struct 字段
說明1 reflect.Value區(qū)分CanSet和Can not Set
所以, 必須要返回成Can set的reflect.Value
如:
s := reflect.ValueOf(&t).Elem()
然后就可以happy的設(shè)值了, 可是不能隨便設(shè)值的, 一個通用的方法就是使用Set(v Value)方法,
說明2 將值轉(zhuǎn)成reflect.Value類型
下面的這段代碼就是轉(zhuǎn)成Value類型
sliceValue := reflect.ValueOf([]int{1, 2, 3}) // 這里將slice轉(zhuǎn)成reflect.Value類型
說明3 reflect.ValueOf 參數(shù)必須是一個 指針 或 interface Elem()才可以正常調(diào)用
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) // 內(nèi)置常用類型的設(shè)值方法 sliceValue := reflect.ValueOf([]int{1, 2, 3}) // 這里將slice轉(zhuǎn)成reflect.Value類型 s.FieldByName("Children").Set(sliceValue)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章
Golang10進(jìn)制轉(zhuǎn)16進(jìn)制的幾種方法代碼示例
這篇文章主要給大家介紹了關(guān)于Golang10進(jìn)制轉(zhuǎn)16進(jìn)制的幾種方法,進(jìn)制轉(zhuǎn)換是Golang的一些基本操作,文中通過實例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-07-07gorm+gin實現(xiàn)restful分頁接口的實踐
本文主要介紹了gorm+gin實現(xiàn)restful分頁接口的實踐,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-12-12golang使用http client發(fā)起get和post請求示例
這篇文章主要介紹了golang使用http client發(fā)起get和post請求示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02基于微服務(wù)框架go-micro開發(fā)gRPC應(yīng)用程序
這篇文章介紹了基于微服務(wù)框架go-micro開發(fā)gRPC應(yīng)用程序的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-07-07深入學(xué)習(xí)Golang并發(fā)編程必備利器之sync.Cond類型
Go?語言的?sync?包提供了一系列同步原語,其中?sync.Cond?就是其中之一。本文將深入探討?sync.Cond?的實現(xiàn)原理和使用方法,幫助大家更好地理解和應(yīng)用?sync.Cond,需要的可以參考一下2023-05-05