Golang?reflect反射的使用實例
首先有一段以下結(jié)構(gòu)體的定義
type User struct { UserName string UserId int `name:"uid"` }
初始化一個結(jié)構(gòu)體的實例
u := User{"octoboy", 101}
獲取字段名
首先獲取變量的Type變量
t := reflect.TypeOf(u)
需要注意的是,如果傳入的u是個指針,比如&User{"octoboy", 101}
if t.Kind() == reflect.Ptr { t = t.Elem() }
這里通過Kind()函數(shù)獲取變量的類型,判斷如果類型為指針 需要使用Elem()獲取指針指向的內(nèi)容。
然后遍歷結(jié)構(gòu)體的字段,獲取其字段名稱
for i := 0; i < t.NumField(); i++ { fmt.Println(t.Field(i).Name) }
輸出結(jié)果:
UserName
UserId
獲取字段類型和值
v := reflect.ValueOf(u) if v.Kind() == reflect.Ptr { //類型為指針 需要取elem v = v.Elem() }
獲取字段的值或者賦值,需要用到ValueOf方法
for i := 0; i < v.NumField(); i++ { //v.Field(i).Int() v.Field(i).String() 都可以把值返回出來,相當于斷言 類型不匹配會直接panic //直接斷成interface 任意類型 fmt.Println(v.Field(i).Interface()) }
輸出結(jié)果:
zyg
101
繼續(xù)輸出成員變量的類型
for i := 0; i < v.NumField(); i++ { fmt.Println(v.Field(i).Kind()) }
輸出結(jié)果:
string
int
設置字段值
靜態(tài)賦值
//設置字段值 va := reflect.ValueOf(&u) //這里必須使用指針 否則后面調(diào)用Set無法使用無地址的值 if va.Kind() == reflect.Ptr { //類型為指針 需要取elem 意為取它指向的內(nèi)容值 va = va.Elem() for i := 0; i < va.NumField(); i++ { //兩種方法取設置字段的值,第二種更為統(tǒng)一 if va.Field(i).Kind() == reflect.String { //重要 如果需要使用set取修改u中的值,需要在ValueOf中傳入u的地址。否則會因為SetString使用了一個不能被尋址的值而造成panic va.Field(i).SetString("octoboy") } if va.Field(i).Kind() == reflect.Int { va.Field(i).Set(reflect.ValueOf(123)) } }
interface切片映射成結(jié)構(gòu)體(動態(tài)賦值)
//練手 values := []interface{}{"octoboy", 123} for i := 0; i < va.NumField(); i++ { if reflect.ValueOf(values[i]).Kind() == va.Field(i).Kind() { va.Field(i).Set(reflect.ValueOf(values[i])) } }
打印以上兩種結(jié)構(gòu)題變量
輸出結(jié)果:
&{octoboy 123}
進階—map映射成結(jié)構(gòu)體
有如下代碼
//練習 把map映射成struct set := map[string]interface{}{ "UserName": "zyg", "UserId": 101, "Age": 19, "Sex": 1, } user := &User{} MapToStruct(set, user) fmt.Println(user)
要求將map映射到user結(jié)構(gòu)題中,即如果User的字段名如存在于map的key中,則將對應的value值賦給user結(jié)構(gòu)題的成員變量
有如下實現(xiàn)
//str類型為interface{} 代表可以傳入任意的結(jié)構(gòu)體 func MapToStruct(m map[string]interface{}, str interface{}) { val := reflect.ValueOf(str) if val.Kind() != reflect.Ptr {//必須是指針 否則無法用Set賦值 panic(any("must be ptr!")) } val = val.Elem() if val.Kind() != reflect.Struct { //指針指向的必須是結(jié)構(gòu)體 panic(any("must be struct")) } for i := 0; i < val.NumField(); i++ { name := val.Type().Field(i).Name //value轉(zhuǎn)type后取字段名稱 if v, ok := m[name]; ok { //如果根據(jù)tag做映射,就使用val.Type().Field(i).Tag.Get("name")作為key if reflect.ValueOf(v).Kind() == val.Field(i).Kind() { val.Field(i).Set(reflect.ValueOf(v)) } } } }
總結(jié)
1.TypeOf 用Name 獲取字段名,也可以用Kind獲取字段的類型;ValueOf 只能用Kind獲取字段的類型。
2.使用reflect.ValueOf(u).Type()轉(zhuǎn)成type,可以與reflect.TypeOf(u)達到一樣的效果
3.如果要使用Set,SetString等方法為其賦值,reflect.ValueOf(u)這里的u必須傳入一個指針,否則無法尋址,會出現(xiàn)panic
4.reflect.ValueOf(u).Interface()可以獲取值,在實際使用過程中往往會將返回結(jié)果斷言成一個接口類型去調(diào)用接口函數(shù)
到此這篇關(guān)于golang reflect反射的使用實例的文章就介紹到這了,更多相關(guān)go reflect反射內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解如何使用go-acme/lego實現(xiàn)自動簽發(fā)證書
這篇文章主要為大家詳細介紹了如何使用?go-acme/lego?的客戶端或庫完成證書的自動簽發(fā),文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下2024-03-03深入解析Go語言編程中slice切片結(jié)構(gòu)
這篇文章主要介紹了Go語言編程中slice切片結(jié)構(gòu),其中Append方法的用法介紹較為詳細,需要的朋友可以參考下2015-10-10golang 并發(fā)安全Map以及分段鎖的實現(xiàn)方法
這篇文章主要介紹了golang 并發(fā)安全Map以及分段鎖的實現(xiàn)方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-03-03go語言題解LeetCode1128等價多米諾骨牌對的數(shù)量
這篇文章主要為大家介紹了go語言題解LeetCode1128等價多米諾骨牌對的數(shù)量示例詳解,2022-12-12