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

