Golang中的深拷貝與淺拷貝使用
一、概念
1、深拷貝(Deep Copy)
拷貝的是數(shù)據(jù)本身,創(chuàng)造一個(gè)樣的新對(duì)象,新創(chuàng)建的對(duì)象與原對(duì)象不共享內(nèi)存,新創(chuàng)建的對(duì)象在內(nèi)存中開(kāi)辟一個(gè)新的內(nèi)存地址,新對(duì)象值修改時(shí)不會(huì)影響原對(duì)象值。既然內(nèi)存地址不同,釋放內(nèi)存地址時(shí),可分別釋放。
值類型的數(shù)據(jù),默認(rèn)全部都是深復(fù)制,Array、Int、String、Struct、Float,Bool。
2、淺拷貝(Shallow Copy)
拷貝的是數(shù)據(jù)地址,只復(fù)制指向的對(duì)象的指針,此時(shí)新對(duì)象和老對(duì)象指向的內(nèi)存地址是一樣的,新對(duì)象值修改時(shí)老對(duì)象也會(huì)變化。釋放內(nèi)存地址時(shí),同時(shí)釋放內(nèi)存地址。
引用類型的數(shù)據(jù),默認(rèn)全部都是淺復(fù)制,Slice,Map。
二、本質(zhì)區(qū)別
是否真正獲?。◤?fù)制)對(duì)象實(shí)體,而不是引用。
三、示例
淺拷貝
等號(hào)賦值
package main import ( "fmt" "reflect" "unsafe" ) func main() { slice1 := []int{1,2,3,4,5} slice2 := slice1 fmt.Println(slice1) fmt.Println(slice2) //同時(shí)改變兩個(gè)數(shù)組 slice1[1]=100 fmt.Println(slice1) fmt.Println(slice2) fmt.Println("切片1指向的底層數(shù)組地址:",(*reflect.SliceHeader)(unsafe.Pointer(&slice1))) fmt.Println("切片2指向的底層數(shù)組地址:",(*reflect.SliceHeader)(unsafe.Pointer(&slice2))) }
輸出信息:
[1 2 3 4 5]
[1 2 3 4 5]
[1 100 3 4 5]
[1 100 3 4 5]
切片1指向的底層數(shù)組地址: &{824634425392 5 5}
切片2指向的底層數(shù)組地址: &{824634425392 5 5}
關(guān)于copy函數(shù):
1.copy只能用于切片,不能用于 map 等任何其他類型。
2.copy返回結(jié)果為一個(gè) int 型值,表示 copy 從原切片src復(fù)制到目的切片的長(zhǎng)度。
使用注意事項(xiàng):
切片 dst 需要先初始化長(zhǎng)度
在使用copy將 src 完全 復(fù)制 到 dst 時(shí),需要初始化目的切片dst的長(zhǎng)度。
1.如果 dst 長(zhǎng)度小于 src 的長(zhǎng)度,則 拷貝src中的部分內(nèi)容;
2.如果大于,則全部拷貝過(guò)來(lái),其余的空間填充該類型的默認(rèn)值;
3.如果相等,剛好不多不少 copy 過(guò)來(lái),所以,通常dst在初始化時(shí)即指定其為src的長(zhǎng)度。
package main import "fmt" func main() { src := []int{1, 2, 3, 5, 6, 7, 8} fmt.Println("src len:", len(src), "src:", src) dst := make([]int, 0) copy(dst, src) fmt.Println("dst len:", len(dst), "dst:", dst) dst1 := make([]int, len(src)/2 ) copy(dst1, src) fmt.Println("dst1 len:", len(dst1), "dst1:", dst1) dst2 := make([]int, len(src)) copy(dst2, src) fmt.Println("dst2 len:", len(dst2), "dst2:", dst2) dst3 := make([]int, len(src) + 2) copy(dst3, src) fmt.Println("dst3 len:", len(dst3), "dst3:", dst3) }
輸出
src len: 7 src: [1 2 3 5 6 7 8]
dst len: 0 dst: []
dst1 len: 3 dst1: [1 2 3]
dst2 len: 7 dst2: [1 2 3 5 6 7 8]
dst3 len: 9 dst3: [1 2 3 5 6 7 8 0 0]
源切片中元素類型為引用類型時(shí),拷貝的是引用
由于copy 函數(shù),拷貝的是切片中的元素,所以如果切片元素的類型是引用類型,那么 copy 的也將是個(gè)引用。
如下面例子,matA 和 matB 地址不一樣,但 matA[0] 和 matB[0] 的地址是一樣的。
func wrongCopyMatrix() { matA := [][]int{ {0, 1, 1, 0}, {0, 1, 1, 1}, {1, 1, 1, 0}, } matB := make([][]int, len(matA)) copy(matB, matA) fmt.Printf("%p, %p\n", matA, matA[0]) // 0xc0000c0000, 0xc0000c2000 fmt.Printf("%p, %p\n", matB, matB[0]) // 0xc0000c0050, 0xc0000c2000 }
如果想 copy 多維切片中的每一個(gè)切片類型的元素,那么就需要將每個(gè)切片元素進(jìn)行 初始化 并 拷貝。注意是兩步:
1.先初始化每維切片,
2.再拷貝。
正確拷貝一個(gè)多維數(shù)組的打開(kāi)方式:
func rightCopyMatrix() { matA := [][]int{ {0, 1, 1, 0}, {0, 1, 1, 1}, {1, 1, 1, 0}, } matB := make([][]int, len(matA)) for i := range matA { matB[i] = make([]int, len(matA[i])) // 注意初始化長(zhǎng)度 copy(matB[i], matA[i]) } fmt.Printf("%p, %p\n", matA, matA[0]) // 0xc00005c050, 0xc000018560 fmt.Printf("%p, %p\n", matB, matB[0]) // 0xc00005c0a0, 0xc0000185c0 }
切片使用copy和等號(hào)復(fù)制的區(qū)別
1.性能方面:copy復(fù)制會(huì)比等號(hào)復(fù)制慢。 2.復(fù)制方式:copy復(fù)制為值復(fù)制,改變?cè)衅闹挡粫?huì)影響新切片。而等號(hào)復(fù)制為指針復(fù)制,改變?cè)衅蛐虑衅紩?huì)對(duì)另一個(gè)產(chǎn)生影響。
深拷貝
(淺)拷貝對(duì)于值類型的話是完全拷貝一份相同的值;而對(duì)于引用類型是拷貝其地址,也就是拷貝的對(duì)象修改引用類型的變量同樣會(huì)影響到源對(duì)象。
對(duì)于深拷貝,任何對(duì)象都會(huì)被完完整整的拷貝一份,拷貝對(duì)象與被拷貝對(duì)象不存在任何聯(lián)系,也就不會(huì)互相影響。
如果你需要拷貝的對(duì)象中沒(méi)有引用類型,那么對(duì)于Golang而言使用淺拷貝就可以了。
基于序列化和反序列化來(lái)實(shí)現(xiàn)對(duì)象的深度拷貝:
import "encoding/gob" func deepCopy(dst, src interface{}) error { var buf bytes.Buffer if err := gob.NewEncoder(&buf).Encode(src); err != nil { return err } return gob.NewDecoder(bytes.NewBuffer(buf.Bytes())).Decode(dst) }
參考:
到此這篇關(guān)于Golang中的深拷貝與淺拷貝使用的文章就介紹到這了,更多相關(guān)Golang 深拷貝與淺拷貝內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
go語(yǔ)言中os包的用法實(shí)戰(zhàn)大全
Go在os中提供了文件的基本操作,包括通常意義的打開(kāi)、創(chuàng)建、讀寫(xiě)等操作,除此以外為了追求便捷以及性能上,Go還在io/ioutil以及bufio提供一些其他函數(shù)供開(kāi)發(fā)者使用,這篇文章主要給大家介紹了關(guān)于go語(yǔ)言中os包用法的相關(guān)資料,需要的朋友可以參考下2024-02-02Go pprof內(nèi)存指標(biāo)含義備忘錄及案例分析
這篇文章主要介紹了Go pprof內(nèi)存指標(biāo)含義備忘錄問(wèn)題,小編特此把問(wèn)題及案例分享到腳本之家平臺(tái)供大家學(xué)習(xí),需要的朋友可以參考下2020-03-03Golang多線程爬蟲(chóng)高效抓取大量數(shù)據(jù)的利器
Golang多線程爬蟲(chóng)是一種高效抓取大量數(shù)據(jù)的利器。Golang語(yǔ)言天生支持并發(fā)和多線程,可以輕松實(shí)現(xiàn)多線程爬蟲(chóng)的開(kāi)發(fā)。通過(guò)使用Golang的協(xié)程和通道,可以實(shí)現(xiàn)爬蟲(chóng)的高效并發(fā)抓取、數(shù)據(jù)處理和存儲(chǔ)2023-05-05go mutex互斥鎖使用Lock和Unlock方法占有釋放資源
Go號(hào)稱是為了高并發(fā)而生的,在高并發(fā)場(chǎng)景下,勢(shì)必會(huì)涉及到對(duì)公共資源的競(jìng)爭(zhēng),當(dāng)對(duì)應(yīng)場(chǎng)景發(fā)生時(shí),我們經(jīng)常會(huì)使用 mutex 的 Lock() 和 Unlock() 方法來(lái)占有或釋放資源,雖然調(diào)用簡(jiǎn)單,但 mutex 的內(nèi)部卻涉及挺多的,本文來(lái)好好研究一下2023-09-09Go 語(yǔ)言入門(mén)學(xué)習(xí)之正則表達(dá)式
這篇文章主要介紹了Go 語(yǔ)言入門(mén)學(xué)習(xí)之正則表達(dá)式,文章基于GO語(yǔ)言的相關(guān)資料展開(kāi)詳細(xì)內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-04-04Go語(yǔ)言defer與return執(zhí)行的先后順序詳解
這篇文章主要為大家介紹了Go語(yǔ)言defer與return執(zhí)行的先后順序詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12