go深度拷贝json版
go中的深度拷贝方式
1.使用json的序列化和反序列化
通过将原来的结构序列化成byte数组,然后将byte数组反序列化到
目标结构的方式来进行深度拷贝。相关代码如下:
package main
import (
"encoding/json"
"fmt"
"reflect"
)
type (
Player struct {
Id int
Level int
Heroes map[int]*Hero
Equips []*Equip
}
Hero struct {
Id int
Level int
Skills []*Skill
}
Equip struct {
Id int
Level int
}
Skill struct {
Id int
Level int
}
)
func NewHero() *Hero {
return &Hero{
Id: 1,
Level: 1,
Skills: append([]*Skill{NewSkill()}, NewSkill(), NewSkill()),
}
}
func NewSkill() *Skill {
return &Skill{1, 1}
}
func NewEquip() *Equip {
return &Equip{1, 1}
}
func NewPlayer() *Player {
return &Player{
Id: 1,
Level: 1,
Heroes: map[int]*Hero{1: NewHero(), 2: NewHero(), 3: NewHero()},
Equips: append([]*Equip{NewEquip()}, NewEquip(), NewEquip()),
}
}
func CopyByJson(dst interface{}, src interface{}) error {
if dst == nil {
return fmt.Errorf("dst cannot be nil")
}
if src == nil {
return fmt.Errorf("src cannot be nil")
}
bytes, err := json.Marshal(src)
if err != nil {
return fmt.Errorf("Unable to marshal src: %s", err)
}
err = json.Unmarshal(bytes, dst)
if err != nil {
return fmt.Errorf("Unable to unmarshal into dst: %s", err)
}
return nil
}
func (self *Hero) Print() {
fmt.Printf("Id=%d, Level=%d\n", self.Id, self.Level)
for _, v := range self.Skills {
fmt.Printf("%v\n", *v)
}
}
func (self *Player) Print() {
fmt.Printf("Id=%d, Level=%d\n", self.Id, self.Level)
for _, v := range self.Heroes {
v.Print()
}
for _, v := range self.Equips {
fmt.Printf("%+v\n", *v)
}
}
func main() {
p1 := NewPlayer()
p2 := new(Player)
CopyByJson(p2, p1)
fmt.Println(reflect.DeepEqual(p1, p2))
p2.Print()
}
output:
true
//对应的benchmark
func BenchmarkCopyByJson(b *testing.B) {
p1 := NewPlayer()
p2 := new(Player)
for i:=0 ; i<b.N ; i++ {
CopyByJson(p2, p1)
}
}
goos: windows
goarch: amd64
pkg: game.lab/go-deepcopy/src/main
100000 21600 ns/op
PASS

浙公网安备 33010602011771号