golang 實現(xiàn)兩個結(jié)構(gòu)體復(fù)制字段
實際工作中可能會有這樣的場景:
兩個結(jié)構(gòu)體(可能類型一樣), 字段名和類型都一樣, 想復(fù)制一個結(jié)構(gòu)體的全部或者其中某幾個字段的值到另一個(即merge操作),
自然想到可以用反射實現(xiàn)
package main import "fmt" import "reflect" // 用b的所有字段覆蓋a的 // 如果fields不為空, 表示用b的特定字段覆蓋a的 // a應(yīng)該為結(jié)構(gòu)體指針 func CopyFields(a interface{}, b interface{}, fields ...string) (err error) { at := reflect.TypeOf(a) av := reflect.ValueOf(a) bt := reflect.TypeOf(b) bv := reflect.ValueOf(b) // 簡單判斷下 if at.Kind() != reflect.Ptr { err = fmt.Errorf("a must be a struct pointer") return } av = reflect.ValueOf(av.Interface()) // 要復(fù)制哪些字段 _fields := make([]string, 0) if len(fields) > 0 { _fields = fields } else { for i := 0; i < bv.NumField(); i++ { _fields = append(_fields, bt.Field(i).Name) } } if len(_fields) == 0 { fmt.Println("no fields to copy") return } // 復(fù)制 for i := 0; i < len(_fields); i++ { name := _fields[i] f := av.Elem().FieldByName(name) bValue := bv.FieldByName(name) // a中有同名的字段并且類型一致才復(fù)制 if f.IsValid() && f.Kind() == bValue.Kind() { f.Set(bValue) } else { fmt.Printf("no such field or different kind, fieldName: %s\n", name) } } return } type S1 struct { Name string Age int } type S2 struct { Name string Age int32 } func main() { s1 := S1{"hello", 22} s2 := S2{"world", 33} fmt.Println(s1, s2) CopyFields(&s1, s2) fmt.Println(s1, s2) }
上述例子輸出為:
{hello 22} {world 33}
no such field or different kind, fieldName: Age
{world 22} {world 33}
可見s2的Name字段值已經(jīng)成功被覆蓋.
而s2中Age字段和s1中Age字段類型不一樣, 會忽略.
其實上面的還可以優(yōu)化, 畢竟int32和int還是可以認為是"一樣"的類型的,
不過思路就是這樣.
補充:golang使用反射將一個結(jié)構(gòu)體的數(shù)據(jù)直接復(fù)制到另一個結(jié)構(gòu)體中(通過相同字段)
看代碼吧~
package main import ( "fmt" "reflect" ) type A struct { Name string Gender string Age int } type B struct { Name string Gender string } //binding type interface 要修改的結(jié)構(gòu)體 //value type interace 有數(shù)據(jù)的結(jié)構(gòu)體 func structAssign(binding interface{}, value interface{}) { bVal := reflect.ValueOf(binding).Elem() //獲取reflect.Type類型 vVal := reflect.ValueOf(value).Elem() //獲取reflect.Type類型 vTypeOfT := vVal.Type() for i := 0; i < vVal.NumField(); i++ { // 在要修改的結(jié)構(gòu)體中查詢有數(shù)據(jù)結(jié)構(gòu)體中相同屬性的字段,有則修改其值 name := vTypeOfT.Field(i).Name if ok := bVal.FieldByName(name).IsValid(); ok { bVal.FieldByName(name).Set(reflect.ValueOf(vVal.Field(i).Interface())) } } } func main() { as := A{} bs := B{Name: "wfy", Gender: "男"} fmt.Println(as) structAssign(&as, &bs) fmt.Println(as) }
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章
詳解Go語言如何實現(xiàn)類似Python中的with上下文管理器
熟悉?Python?的同學(xué)應(yīng)該知道?Python?中的上下文管理器非常好用,那么在?Go?中是否也能實現(xiàn)上下文管理器呢,下面小編就來和大家仔細講講吧2023-07-07Go語言正則表達式用法實例小結(jié)【查找、匹配、替換等】
這篇文章主要介紹了Go語言正則表達式用法,結(jié)合實例形式分析了Go語言基于正則實現(xiàn)查找、匹配、替換等基本操作的實現(xiàn)技巧,需要的朋友可以參考下2017-01-01Golang微服務(wù)框架Kratos實現(xiàn)Kafka消息隊列的方法
消息隊列是大型分布式系統(tǒng)不可缺少的中間件,也是高并發(fā)系統(tǒng)的基石中間件,所以掌握好消息隊列MQ就變得極其重要,在本文當(dāng)中,您將了解到:什么是消息隊列?什么是Kafka?怎樣在微服務(wù)框架Kratos當(dāng)中應(yīng)用Kafka進行業(yè)務(wù)開發(fā),需要的朋友可以參考下2023-09-09golang-gin-mgo高并發(fā)服務(wù)器搭建教程
這篇文章主要介紹了golang-gin-mgo高并發(fā)服務(wù)器搭建教程,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12