前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >golang deepcopy_mongodb主从复制原理

golang deepcopy_mongodb主从复制原理

作者头像
全栈程序员站长
发布2022-11-11 14:55:15
2440
发布2022-11-11 14:55:15
举报
文章被收录于专栏:全栈程序员必看

大家好,又见面了,我是你们的朋友全栈君。

Go语言中所有赋值操作都是值传递,如果结构中不含指针,则直接赋值就是深度拷贝;如果结构中含有指针(包括自定义指针,以及切片,map等使用了指针的内置类型),则数据源和拷贝之间对应指针会共同指向同一块内存,这时深度拷贝需要特别处理。目前,有三种方法,一是用gob序列化成字节序列再反序列化生成克隆对象;二是先转换成json字节序列,再解析字节序列生成克隆对象;三是针对具体情况,定制化拷贝。前两种方法虽然比较通用但是因为使用了reflex反射,性能比定制化拷贝要低出2个数量级,所以在性能要求较高的情况下应该尽量避免使用前两者。

结论数据:

执行一次的时间

gob time:454µs json time:170µs custom time:2µs

测试代码如下:

代码语言:javascript
复制
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)
}

运行输出:

代码语言:javascript
复制
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

相关文章:

《Go语言:append函数源码学习及切片深度拷贝问题》

《Go语言切片拷贝时容量对于效率的影响》

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/186667.html原文链接:https://javaforall.cn

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022年10月2日 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
文件存储
文件存储(Cloud File Storage,CFS)为您提供安全可靠、可扩展的共享文件存储服务。文件存储可与腾讯云服务器、容器服务、批量计算等服务搭配使用,为多个计算节点提供容量和性能可弹性扩展的高性能共享存储。腾讯云文件存储的管理界面简单、易使用,可实现对现有应用的无缝集成;按实际用量付费,为您节约成本,简化 IT 运维工作。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档