大家好,又见面了,我是你们的朋友全栈君。
Go语言中所有赋值操作都是值传递,如果结构中不含指针,则直接赋值就是深度拷贝;如果结构中含有指针(包括自定义指针,以及切片,map等使用了指针的内置类型),则数据源和拷贝之间对应指针会共同指向同一块内存,这时深度拷贝需要特别处理。目前,有三种方法,一是用gob序列化成字节序列再反序列化生成克隆对象;二是先转换成json字节序列,再解析字节序列生成克隆对象;三是针对具体情况,定制化拷贝。前两种方法虽然比较通用但是因为使用了reflex反射,性能比定制化拷贝要低出2个数量级,所以在性能要求较高的情况下应该尽量避免使用前两者。
结论数据:
执行一次的时间
gob time:454µs json time:170µs custom time:2µs
测试代码如下:
package main
import (
"bytes"
"encoding/gob"
"encoding/json"
"fmt"
"time"
)
type AuthorInfo struct {
Name string `json:name`
Age int `json:age`
Country *int `json:country`
}
type Book struct {
Title string `json:title`
Author AuthorInfo `json:author`
Year int `json:year`
Category []string `json:category`
Price map[string]string `json:price`
}
func DeepCopyByGob(dst, src interface{}) error {
var buffer bytes.Buffer
if err := gob.NewEncoder(&buffer).Encode(src); err != nil {
return err
}
return gob.NewDecoder(&buffer).Decode(dst)
}
func DeepCopyByJson(src []Book) (*[]Book, error) {
var dst = new([]Book)
b, err := json.Marshal(src)
if err != nil {
return nil, err
}
err = json.Unmarshal(b, dst)
return dst, err
}
func DeepCopyByCustom(src []Book) []Book {
dst := make([]Book, len(src))
for i, book := range src {
tmpbook := Book{}
tmpbook.Title = book.Title
tmpbook.Year = book.Year
tmpbook.Author = AuthorInfo{}
tmpbook.Author.Name = book.Author.Name
tmpbook.Author.Age = book.Author.Age
tmpbook.Author.Country = new(int)
*tmpbook.Author.Country = *book.Author.Country
tmpbook.Category = make([]string, len(book.Category))
for index, category := range book.Category {
tmpbook.Category[index] = category
}
tmpbook.Price = make(map[string]string)
for k, v := range book.Price {
tmpbook.Price[k] = v
}
dst[i] = tmpbook
}
return dst
}
func check(err error){
if err != nil{
panic(err)
}
}
func print(name string, books []Book){
for index,book := range books{
fmt.Printf("%s[%d]=%v country=%d\n", name, index, book, *book.Author.Country)
}
}
func main() {
//初始化源Book切片
books := make([]Book, 1)
country := 1156
author := AuthorInfo{"David", 38, &country}
price := make(map[string]string)
price["Europe"] = "$56"
books[0] = Book{"Tutorial", author, 2020, []string{"math", "art"}, price}
print("books",books)
var err error
var start time.Time
//Gob拷贝
start = time.Now()
booksCpy := make([]Book, 1)
err = DeepCopyByGob(&booksCpy, books)
fmt.Printf("\ngob time:%v\n", time.Now().Sub(start))
check(err)
*booksCpy[0].Author.Country = 1134
booksCpy[0].Category[0] = "literature"
booksCpy[0].Price["America"] = "$250"
print("booksCpy",booksCpy)
print("books",books)
//JSON拷贝
start = time.Now()
booksCpy2, err_json := DeepCopyByJson(books)
fmt.Printf("\njson time:%v\n", time.Now().Sub(start))
check(err_json)
*(*booksCpy2)[0].Author.Country = 1135
(*booksCpy2)[0].Category[0] = "science"
(*booksCpy2)[0].Price["Canada"] = "$150"
print("(*booksCpy2)",*booksCpy2)
print("books",books)
//定制拷贝
start = time.Now()
booksCpy3 := DeepCopyByCustom(books)
fmt.Printf("\ncustom time:%v\n", time.Now().Sub(start))
*booksCpy3[0].Author.Country = 1136
booksCpy3[0].Category[0] = "geometry"
booksCpy3[0].Price["Africa"] = "$34"
print("booksCpy3",booksCpy3)
print("books",books)
}
运行输出:
books[0]={Tutorial {David 38 0xc000016178} 2020 [math art] map[Europe:$56]} country=1156
gob time:454.117µs
booksCpy[0]={Tutorial {David 38 0xc0000165d8} 2020 [literature art] map[America:$250 Europe:$56]} country=1134
books[0]={Tutorial {David 38 0xc000016178} 2020 [math art] map[Europe:$56]} country=1156
json time:170.338µs
(*booksCpy2)[0]={Tutorial {David 38 0xc000016878} 2020 [science art] map[Canada:$150 Europe:$56]} country=1135
books[0]={Tutorial {David 38 0xc000016178} 2020 [math art] map[Europe:$56]} country=1156
custom time:2.165µs
booksCpy3[0]={Tutorial {David 38 0xc0000168c8} 2020 [geometry art] map[Africa:$34 Europe:$56]} country=1136
books[0]={Tutorial {David 38 0xc000016178} 2020 [math art] map[Europe:$56]} country=1156
相关文章:
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/186667.html原文链接:https://javaforall.cn