Go語言基礎(chǔ)反射示例詳解
概述
在程序運(yùn)行期對程序動態(tài)的進(jìn)行訪問和修改
reflect godoc: https://golang.org/pkg/reflect/
reflect包有兩個數(shù)據(jù)類型:
Type:數(shù)據(jù)類型 【reflect.TypeOf():是獲取Type的方法】
Value:值的類型【reflect.ValueOf():是獲取Value的方法】
語法
一、基本操作
獲取變量類型
func TypeOf(i interface{}) Type //Type是interface{}的別名
例子
reflect.TypeOf(10) //int reflect.TypeOf(struct{ age int }{10}) //struct { age int }
獲取變量的種類
reflect.TypeOf(struct{ age int }{10}).Kind() //reflect.Struct reflect.ValueOf("hello word").Kind() //reflect.String
獲取變量值
func ValueOf(i interface{}) Value //value是struct {}別名
例子
reflect.ValueOf("hello word") //hello word reflect.ValueOf(struct{ age int }{10}) //{10}
二、修改目標(biāo)對象
修改普通類型
str := "hello word" reflect.ValueOf(&str).Elem().SetString("張三")
修改結(jié)構(gòu)體
//第一步:ValueOf():傳入一個變量的地址,返回是變量的地址 Elem():返回的是變量的原始值 elem:=reflect.ValueOf(&變量名).Elem() //第二步 FieldByName():傳入結(jié)構(gòu)體字段名稱 SetString():傳入你要修改的變量值 elem.FieldByName("Name").SetString("李四")
//定義一個User結(jié)構(gòu)體 type User struct { Name string Age int } user := User{Name: "張三", Age: 10} //Elem() 獲取user原始的值 elem := reflect.ValueOf(&user).Elem() //FieldByName() 通過Name返回具有給定名稱的結(jié)構(gòu)字段 通過SetString 修改原始的值 elem.FieldByName("Name").SetString("李四") elem.FieldByName("Age").SetInt(18)
三、動態(tài)調(diào)用方法
無參方法
//MethodByName():傳方法名,方法名必須大小 Call():方法的形參 reflect.ValueOf(變量名).MethodByName(方法名).Call([]reflect.Value{}) reflect.ValueOf(變量名).MethodByName(方法名).Call(make([]reflect.Value, 0))
type User struct { Name string `json:"name" name:"張三"` Age int } func (_ User) Say() { fmt.Println("user 說話") } user := User{Name: "張三", Age: 10} reflect.ValueOf(&user).MethodByName("Say").Call([]reflect.Value{}) reflect.ValueOf(user).MethodByName("Say").Call(make([]reflect.Value, 0))
有參方法
reflect.ValueOf(變量名).MethodByName(方法名).Call([]reflect.Value{reflect.ValueOf("該說話了"), reflect.ValueOf(1)})
type User struct { Name string `json:"name" name:"張三"` Age int } func (_ User) Say() { fmt.Println("user 說話") } user := User{Name: "張三", Age: 10} reflect.ValueOf(user).MethodByName("SayContent").Call([]reflect.Value{reflect.ValueOf("該說話了"), reflect.ValueOf(1)})
總結(jié)
反射調(diào)用struct的方法必須是公有的
反射調(diào)用無參方法時必修傳 nil 或者 []reflect.Value{}
示例
package main import ( "fmt" "reflect" ) func main() { //1. 獲取變量類型 fmt.Println("獲取變量類型") fmt.Println(reflect.TypeOf(10)) //int fmt.Println(reflect.TypeOf(10.0)) //float64 fmt.Println(reflect.TypeOf(struct{ age int }{10})) //struct { age int } fmt.Println(reflect.TypeOf(map[string]string{"a": "a"})) //map[string]string fmt.Println("") //2. 獲取變量值 fmt.Println("獲取變量值") fmt.Println(reflect.ValueOf("hello word")) //hello word fmt.Println(reflect.ValueOf(struct{ age int }{10})) //{10} fmt.Println(reflect.TypeOf(struct{ age int }{10}).Kind()) //struct //類型判斷 if t := reflect.TypeOf(struct{ age int }{10}).Kind(); t == reflect.Struct { fmt.Println("是結(jié)構(gòu)體") } else { fmt.Println("不是結(jié)構(gòu)體") } //修改目標(biāo)對象 str := "hello word" //普通變量修改 reflect.ValueOf(&str).Elem().SetString("張三") fmt.Println(str) //結(jié)構(gòu)體變量修改 user := User{Name: "張三", Age: 10} //Elem() 獲取user原始的值 elem := reflect.ValueOf(&user).Elem() //FieldByName() 通過Name返回具有給定名稱的結(jié)構(gòu)字段 通過SetString 修改原始的值 elem.FieldByName("Name").SetString("李四") elem.FieldByName("Age").SetInt(18) fmt.Println(user) //獲取結(jié)構(gòu)體的標(biāo)簽的值 fmt.Println(reflect.TypeOf(&user).Elem().Field(0).Tag.Get("name")) //調(diào)用無參方法 reflect.ValueOf(&user).MethodByName("Say").Call([]reflect.Value{}) reflect.ValueOf(user).MethodByName("Say").Call(make([]reflect.Value, 0)) //調(diào)用有參方法 reflect.ValueOf(user).MethodByName("SayContent").Call([]reflect.Value{reflect.ValueOf("該說話了"), reflect.ValueOf(1)}) //調(diào)用本地的方法 reflect.ValueOf(Hello).Call([]reflect.Value{}) reflect.ValueOf(Hello).Call(nil) fmt.Printf("%#v\n", reflect.TypeOf(user).Field(0)) } func Hello() { fmt.Println("hello") } type Person struct { Name string } type User struct { Person // //反射會將匿名字段作為一個獨(dú)立字段來處理 Name string `json:"name" name:"張三"` Age int } func (_ User) Say() { fmt.Println("user 說話") } func (_ User) SayContent(content string, a int) { fmt.Println("user", content, a) }
以上就是Go語言基礎(chǔ)反射示例詳解的詳細(xì)內(nèi)容,更多關(guān)于Go語言反射的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Go項(xiàng)目與Docker結(jié)合實(shí)現(xiàn)高效部署深入探究
在現(xiàn)代軟件開發(fā)中,使用Docker部署應(yīng)用程序已經(jīng)成為一種標(biāo)準(zhǔn)實(shí)踐,本文將深入探討如何將Go項(xiàng)目與Docker結(jié)合,實(shí)現(xiàn)高效、可靠的部署過程,通過詳細(xì)的步驟和豐富的示例,你將能夠迅速掌握這一流程2023-12-12go sync Once實(shí)現(xiàn)原理示例解析
這篇文章主要為大家介紹了go sync Once實(shí)現(xiàn)原理示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01golang中的select關(guān)鍵字用法總結(jié)
這篇文章主要介紹了golang中的select關(guān)鍵字用法總結(jié),本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-06-06在Golang中實(shí)現(xiàn)RSA算法的加解密操作詳解
RSA 是一種非對稱加密算法,廣泛使用于數(shù)據(jù)的安全傳輸,crypto/rsa 是 Golang 中實(shí)現(xiàn)了 RSA 算法的一個標(biāo)準(zhǔn)庫,提供了生成公私鑰對、加解密數(shù)據(jù)、簽名和驗(yàn)簽等功能,本文給大家介紹了在Golang中實(shí)現(xiàn)RSA算法的加解密操作,需要的朋友可以參考下2023-12-12???????Golang實(shí)現(xiàn)RabbitMQ中死信隊(duì)列幾種情況
本文主要介紹了???????Golang實(shí)現(xiàn)RabbitMQ中死信隊(duì)列幾種情況,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03Go實(shí)現(xiàn)SMTP郵件發(fā)送訂閱功能(包含163郵箱、163企業(yè)郵箱、谷歌gmail郵箱)
這篇文章給大家介紹了Go實(shí)現(xiàn)SMTP郵件發(fā)送訂閱功能(包含163郵箱、163企業(yè)郵箱、谷歌gmail郵箱),需求很簡單,就是用戶輸入自己的郵箱后,使用官方郵箱給用戶發(fā)送替郵件模版,文中有詳細(xì)的代碼示例供大家參考,需要的朋友可以參考下2023-10-10Golang使用切片實(shí)現(xiàn)單鏈表的示例代碼
單鏈表(Single?Linked?List)是鏈表數(shù)據(jù)結(jié)構(gòu)的一種實(shí)現(xiàn)方式,它包含一系列節(jié)點(diǎn)(Node),每個節(jié)點(diǎn)都包含一個數(shù)據(jù)域和一個指向下一個節(jié)點(diǎn)的指針,本文給大家介紹了Golang使用切片實(shí)現(xiàn)單鏈表的操作,需要的朋友可以參考下2024-05-05Go語言異常處理error、panic、recover的使用
GO語言中引入的異常的處理方式為error、panic、recover ,本文主要介紹了Go語言異常處理error、panic、recover的使用,感興趣的可以了解一下2024-08-08