国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

Go語言Append缺陷引發(fā)的深度拷貝討論

開發(fā) 后端
今天的文章從我周六加班改的一個bug引入,上下文是在某個struct中有個Labels切片,在組裝數(shù)據(jù)的時候需要為其加上配置變量中的標(biāo)簽。

[[414910]]

看完蘇炳添進(jìn)入總決賽,看得我熱血沸騰的,上廁所都不敢耽擱超過 5 分鐘。

這歷史性的一刻,讓本決定休息的我,垂死病中驚坐起,開始肝文章。

  • 引子
  • 何謂淺?何謂深?
  • 深拷貝的四種方式
  • 手寫拷貝函數(shù)
  • json序列化反序列化
  • gob序列化反序列化
  • 基準(zhǔn)測試(性能測試)
  • 小結(jié)

引子

今天的文章從我周六加班改的一個bug引入,上下文是在某個struct中有個Labels切片,在組裝數(shù)據(jù)的時候需要為其加上配置變量中的標(biāo)簽。

大家看看會出現(xiàn)什么問題。

  1. for i := range m{ 
  2.     m[i].Labels = append(r.Config.Relabel, m[i].Labels...) 
  3.     ... 

debug發(fā)現(xiàn),i=0時正常,但第二次乃至第n次會不斷變更之前m[?].Labels的內(nèi)容。

看了append的源碼,原來當(dāng)容量足夠的時候,append會把數(shù)據(jù)直接添加到第一個參數(shù)的切片里。

改為如下代碼,調(diào)換下了位置,一切正常了。

  1. m[i].Labels = append(m[i].Labels,r.Config.Relabel...) 

這是一個隱含的陷阱,在 go 語言中賦值拷貝往往都是淺拷貝,開發(fā)者很容易不小心忽視這一點,導(dǎo)致這種無法預(yù)料的問題出現(xiàn),以后要多多注意了。

借由這個問題以及上一篇文章的作業(yè)中,提到的深度拷貝問題展開今天的文章。

何謂淺?何謂深?

我多年以前是做c++的,它的對象拷貝是淺拷貝,原理是調(diào)用了默認(rèn)的拷貝構(gòu)造函數(shù),需要人為的重寫,進(jìn)行拷貝的過程,特別是指針需要謹(jǐn)慎的生成的釋放,來避免內(nèi)存泄露的發(fā)生。

后來接觸了Python 發(fā)現(xiàn)深淺拷貝的問題在后端語言中都是存在的,Go 也不例外。

淺拷貝對于值類型是完全拷貝一份,而對于引用類型是拷貝其地址。也就是拷貝的對象修改引用類型的變量同樣會影響到源對象。

這就是為什么channel在做參數(shù)傳遞的時候,向內(nèi)部寫入內(nèi)容,接收端可以成功收到的原因。

在Go中,指針、slice、channel、interface、map、函數(shù)都是淺拷貝。最容易出問題的就是指針、切片、map這三種類型。

方便的點是作為參數(shù)傳遞不需要取地址可以直接修改其內(nèi)容,只要函數(shù)內(nèi)部不出現(xiàn)覆蓋就不需要返回值。

但作為結(jié)構(gòu)體中的成員變量,在拷貝結(jié)構(gòu)體后問題就暴露出來了。修改一處導(dǎo)致另一處也變了。

深拷貝的四種方式

有一次和女朋友聊到深拷貝的問題,她告訴我最方便的深拷貝方法就是序列化為json再反序列化。

我聽到這種方案,頓時驚為天人,確實挺省事的,但由于序列化會用到反射,效率自然不會太高。

深拷貝有四種方式

  • 1、手寫拷貝函數(shù)
  • 2、json序列化反序列化
  • 3、gob序列化反序列化
  • 4、使用反射

github上的開源庫,大多基于 1、4 兩種方式做的優(yōu)化。這里的反射方法后面再做討論。

我的github https://github.com/minibear2333/ 后續(xù)會專門寫一個組件,提供深度拷貝的各種現(xiàn)成的方式。

手寫拷貝函數(shù)

定義一個包含切片、字典、指針的結(jié)構(gòu)體。

  1. type Foo struct { 
  2.  List   []int 
  3.  FooMap map[string]string 
  4.  intPtr *int 

手動拷貝函數(shù),把它取名為Duplicate

  1. func (f *Foo) Duplicate() Foo { 
  2.  var tmp = Foo{ 
  3.   List:   make([]int, 0, len(f.List)), 
  4.   FooMap: make(map[string]string), 
  5.   intPtr: new(int), 
  6.  } 
  7.  copy(tmp.List, f.List) 
  8.  for i := range f.FooMap { 
  9.   tmp.FooMap[i] = f.FooMap[i] 
  10.  } 
  11.  if f.intPtr != nil { 
  12.   *tmp.intPtr = *f.intPtr 
  13.  } else { 
  14.   tmp.intPtr = nil 
  15.  } 
  16.  return tmp 
  • 函數(shù)內(nèi)部初始化結(jié)構(gòu)體
  • copy是標(biāo)準(zhǔn)庫自帶的拷貝函數(shù)
  • map只能range來拷貝,這里map為nil不會報錯
  • 指針使用前必須判空,為指針的指向賦值,而不能覆蓋指針地址

測試

  1. func main() { 
  2.  var a = 1 
  3.  var t1 = Foo{intPtr: &a} 
  4.  t2 := t1.Duplicate() 
  5.  a = 2 
  6.  fmt.Println(*t1.intPtr) 
  7.  fmt.Println(*t2.intPtr) 

輸出說明深拷貝成功

json序列化反序列化

這種方式完成深度拷貝非常簡單,但必須結(jié)構(gòu)體加上注解,而且不允許出現(xiàn)私有字段

  1. type Foo struct { 
  2.  List   []int             `json:"list"
  3.  FooMap map[string]string `json:"foo_map"
  4.  IntPtr *int              `json:"int_ptr"

提供一個直接的方案

  1. func DeepCopyByJson(dst, src interface{}) error { 
  2.  b, err := json.Marshal(src) 
  3.  if err != nil { 
  4.   return err 
  5.  } 
  6.  err = json.Unmarshal(b, dst) 
  7.  
  8.  return err 
  • 其中src和dst是同一種結(jié)構(gòu)體類型
  • dst使用時必須取地址,因為要給地址指向的數(shù)據(jù)變更新值

用法,我省略了錯誤處理

  1. a = 3 
  2. t1 = Foo{IntPtr: &a} 
  3. t2 = Foo{} 
  4. _ = DeepCopyByJson(&t2, t1) 
  5. fmt.Println(*t1.IntPtr) 
  6. fmt.Println(*t2.IntPtr) 

輸出

gob序列化反序列化

這是一種標(biāo)準(zhǔn)庫提供的編碼方法,類似于protobuf,Gob(即 Go binary 的縮寫)。類似于 Python 的pickle和 Java 的Serialization。

在發(fā)送端編碼,接收端解碼。

  1. func DeepCopyByGob(dst, src interface{}) error { 
  2.  var buffer bytes.Buffer 
  3.  if err := gob.NewEncoder(&buffer).Encode(src); err != nil { 
  4.   return err 
  5.  } 
  6.  return gob.NewDecoder(&buffer).Decode(dst) 

用法

  1. func DeepCopyByGob(dst, src interface{}) error { 
  2.  var buffer bytes.Buffer 
  3.  if err := gob.NewEncoder(&buffer).Encode(src); err != nil { 
  4.   return err 
  5.  } 
  6.  return gob.NewDecoder(&buffer).Decode(dst) 

輸出

基準(zhǔn)測試(性能測試)

這三種方式我分別寫了基準(zhǔn)測試的測試用例,go會自動反復(fù)調(diào)用,直到測算出一個合理的時間范圍。

基準(zhǔn)測試代碼,這里僅寫一個,其他兩個函數(shù)的測試方式類似:

  1. func BenchmarkDeepCopyByJson(b *testing.B) { 
  2.  b.StopTimer() 
  3.  var a = 1 
  4.  var t1 = Foo{IntPtr: &a} 
  5.  t2 := Foo{} 
  6.  b.StartTimer() 
  7.  for i := 0; i < b.N; i++ { 
  8.   _ = DeepCopyByJson(&t2, t1) 
  9.  } 

運行測試

  1. $ go test -test.bench=. -cpu=1,16  -benchtime=2s 
  2. goos: darwin 
  3. goarch: amd64 
  4. pkg: my_copy 
  5. cpu: Intel(R) Core(TM) i5-8257U CPU @ 1.40GHz 
  6. BenchmarkFoo_Duplicate          35887767                62.64 ns/op 
  7. BenchmarkFoo_Duplicate-16       37554250                62.56 ns/op 
  8. BenchmarkDeepCopyByGob            104292             22941 ns/op 
  9. BenchmarkDeepCopyByGob-16         103060             23049 ns/op 
  10. BenchmarkDeepCopyByJson          2052482              1171 ns/op 
  11. BenchmarkDeepCopyByJson-16       2057090              1175 ns/op 
  12. PASS 
  13. ok      my_copy 17.166s 
  • 在mac環(huán)境下單核和多核并沒有明顯差異
  • 運行速度快慢,手動拷貝方式 > json > gob
  • 拷貝方式都相差了 2 個數(shù)量級

小結(jié)

如果是偶爾使用的程序可以使用json序列化反序列化的方式進(jìn)行拷貝,但是除了慢以外還有一個缺陷,就是無法拷貝私有成員變量。

如果是頻繁拷貝的程序,建議使用手動拷貝方式進(jìn)行拷貝,而且可以定制化拷貝的過程。甚至可以完成不同結(jié)構(gòu)體之間,字段細(xì)微差異的定制化需求。

PS:內(nèi)置copy和reflect.copy都只支持切片或數(shù)組的拷貝,內(nèi)置copy速度是反射方式的兩倍以上。

拓展資料

  • Go 語言使用 Gob 傳輸數(shù)據(jù) http://c.biancheng.net/view/4597.html)
  • 內(nèi)建copy函數(shù)和reflect.Copy函數(shù)的區(qū)別 https://studygolang.com/topics/13523/comment/43357
  • 基準(zhǔn)測試 https://segmentfault.com/a/1190000016354758

 本文轉(zhuǎn)載自微信公眾號「機智的程序員小熊」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系機智的程序員小熊公眾號。

 

責(zé)任編輯:武曉燕 來源: 機智的程序員小熊
相關(guān)推薦

2024-03-08 09:25:18

.NET深度拷貝淺拷貝

2025-12-02 00:10:00

Go編譯器函數(shù)

2021-06-08 07:45:44

Go語言優(yōu)化

2021-07-28 07:53:21

Go語言拷貝

2012-04-09 09:53:56

2013-07-24 15:29:24

思科Sourcefire思科收購Sourcef

2021-07-08 23:53:44

Go語言拷貝

2023-12-15 14:38:00

GoRust編程語言

2020-05-07 11:00:24

Go亂碼框架

2010-03-01 16:38:08

Linux分區(qū)方案

2011-11-02 09:04:15

Node.js

2009-12-29 17:21:24

Ubuntu 8.04

2009-12-29 16:59:17

Ubuntu Vist

2021-05-12 08:53:54

Go語言調(diào)度

2009-11-27 16:07:10

2010-07-20 10:14:22

蘋果天線門

2014-05-29 10:54:20

C++構(gòu)造函數(shù)

2025-06-04 02:15:55

Go語言Slice

2012-10-08 09:25:59

GoGo語言開發(fā)語言

2018-03-12 22:13:46

GO語言編程軟件
點贊
收藏

51CTO技術(shù)棧公眾號

欧美性xxx| 99久久99热这里只有精品| 国产激情一区二区三区四区| 97精品欧美一区二区三区| 免费福利在线观看| 久久久久久免费| 一本色道久久99精品综合| 精品国产中文字幕第一页| 永久免费精品影视网站| dy888亚洲精品一区二区三区| 亚洲午夜电影网| 精品人妻一区二区三区四区在线 | 午夜日韩av| 久久99精品视频一区97| 中文在线免费| 色天使久久综合网天天| 日日碰狠狠丁香久燥| 精品伊人久久久久7777人| 成人欧美一区二区三区在线观看 | 日韩不卡视频在线观看| 国产亚洲成aⅴ人片在线观看| 俺也去精品视频在线观看| 香蕉成人影院| 污污网址在线观看| 精品176二区| 久久福利毛片| 亚洲国产精品久久艾草纯爱| 国产精品久久久对白| 国语对白在线视频| 婷婷亚洲精品| 国产无遮挡一区二区三区毛片日本| 国产精品video| 国产精久久久| 日本黄网站色大片免费观看| 亚洲一二三四在线观看| 1024亚洲| 亚洲影院久久精品| 久久精品国产68国产精品亚洲| 鲁丝一区鲁丝二区鲁丝三区| 欧美+日本+国产+在线a∨观看| 国产日韩av在线播放| 日韩av网站在线观看| av成人动漫| 综合久久国产九一剧情麻豆| 成年人免费网站| xxx欧美精品| 成人高清av| 欧美成人福利在线观看| 久久免费视频在线观看| av中文字幕一区二区三区| 国产精品久久久久久av下载红粉| 日韩不卡免费视频| xxxx影院| 久久久电影免费观看完整版| 亚洲亚洲人成综合网络| 另类春色校园亚洲| 97超级碰碰| 免费视频一区| 潘金莲一级淫片aaaaa免费看| 91成人在线精品| 国产免费播放一区二区| 免费黄色在线| 国产精品免费久久久| 久久久久久久综合| 欧美影院精品| 波多野结衣在线| 日本特级黄色大片| 尤物在线观看一区| 中文在线观看免费| 日韩精品无码一区二区三区| 久久国产免费| 台湾色综合娱乐中文网| 2018av男人天堂| 亚洲精品影视| 亚洲日本理论电影| 久久久蜜臀国产一区二区| 原千岁中文字幕| 欧美色综合天天久久综合精品| heyzo在线欧美播放| 欧美乱大交xxxxx另类电影| 天天综合亚洲| 欧洲精品视频在线| 国产精品久线在线观看| 国产视频网站在线| 在线观看国产精品淫| 国产亚洲电影| 亚洲精品一区二区毛豆| 中文在线一区二区| 久草中文在线观看| 欧美激情手机在线视频 | 日韩电影中文字幕在线| 日韩精品久久久久久久软件91| 国产日韩欧美日韩大片| 麻豆极品一区二区三区| 免费看成人a| 亚洲护士老师的毛茸茸最新章节 | 色呦呦网站一区| a级片在线免费观看| 欧美最猛性xxxxx亚洲精品| 奇米色777欧美一区二区| 国产高潮av| 日韩电影中文字幕av| 成人激情电影在线| 一卡二卡三卡视频| 欧美三级中文字幕在线观看| www一区二区三区| 精品蜜桃一区二区三区| 国产精品国产自产拍在线| av岛国在线| 波多野结衣一区二区三区在线观看 | 成人爽a毛片免费啪啪动漫| 久久久亚洲精选| 奇米影视一区二区三区| 中文字幕电影在线| 欧美日韩999| 免费在线看成人av| 免费在线观看麻豆视频 | 在线天堂一区av电影| 亚洲成人免费观看| 欧美videos粗暴| 日韩.欧美.亚洲| 黑人巨大精品欧美一区免费视频 | 国产啪精品视频网站| 91一区二区三区在线播放| 欧美人与性动交α欧美精品济南到| 国产欧美精品在线播放| 亚洲欧洲精品一区二区精品久久久 | 国产日产亚洲精品| 国产欧美精品在线观看| 香蕉伊大人中文在线观看| 国产乱码精品一区二区三区中文 | 亚洲四区在线观看| 九九热线视频只有这里最精品| 韩国一区二区三区美女美女秀 | 亚洲综合社区网| 综合久久国产九一剧情麻豆| 欧美暴力调教| 亚洲一区二区在线免费观看| 欧美电影一区二区| 国产精品分类| 国产一级免费在线观看| 国产色综合天天综合网| 亚洲国产欧美在线| 国产亚洲一卡2卡3卡4卡新区| 国产三级三级看三级| 久久久999成人| 成人性生交大片免费| 婷婷六月国产精品久久不卡| 在线观看免费黄色片| 日韩av影视综合网| 免费在线看一区| 忘忧草在线日韩www影院| 正义之心1992免费观看全集完整版| 日韩免费性生活视频播放| 久久精品一本| 欧美videos另类精品| 中文字幕久久一区| 欧美日韩在线观看首页| 天天爽人人爽夜夜爽| 欧美极品欧美精品欧美| 国产不卡视频在线| 久久精品99久久久久久久久| 亚洲v中文字幕| 麻豆精品新av中文字幕| 午夜在线a亚洲v天堂网2018| 国产区美女在线| 尤蜜粉嫩av国产一区二区三区| 亚洲free性xxxx护士白浆| 日韩精品在线观看一区| 亚洲色图视频网| 青青草97国产精品免费观看 | 邻家有女韩剧在线观看国语| 国产精品久久久亚洲| 亚洲在线观看免费| 色喇叭免费久久综合网| 欧洲一区av| 久久久国产精品一区二区三区| 欧美一区二区三区喷汁尤物| 狠狠v欧美v日韩v亚洲ⅴ| 欧美97人人模人人爽人人喊视频| 日本久久久久久久久久久久| 国产精欧美一区二区三区| 久久99成人| 精品国产伦一区二区三区观看体验| 日本最新不卡在线| 三上悠亚激情av一区二区三区 | 精品成人免费一区二区在线播放| 欧美在线一区视频| 亚州欧美日韩中文视频| 一区二区高清在线| 一区福利视频| 精品免费av在线| 日本一二三区视频免费高清| 国产精品视频500部| 亚洲欧美三级在线| 中文字幕二三区不卡| 欧美日韩四区| 伊人久久精品| 免费在线一级视频| 中文字幕成人一区|