Golang中深拷貝與淺拷貝詳解

什麼是深拷貝?

深拷貝(Deep Copy)是指原對象與拷貝的新對象互相獨立,對其中任何一個對象的改動都不會對另外一個對象造成影響。值類型的數據默認是深拷貝,例如array、int、string、struct、float和bool類型。

什麼是淺拷貝?

淺拷貝(Shallow Copy)是指將一個對象的一部分復制到另一個對象中,使用指針來引用原始對象,從而實現對原始對象的部分復制。此時新對象和老對象指向的內存地址是一樣的,修改新對象值後老對象值也會變化。引用類型的數據默認是淺拷貝,例如slice和map。

示例代碼

對於引用類型對象來說,使用等號賦值就是淺拷貝,看如下代碼片段:

package main

import (
	"fmt"
	"reflect"
	"unsafe"
)

func main() {
	slice1 := []int{1, 2, 3, 4, 5, 6}

	//slice2是slice1的淺拷貝
	slice2 := slice1
	fmt.Println(slice1)
	fmt.Println(slice2)

	//修改slice1的值,slice2的值也會發生改變
	slice1[1] = 100
	fmt.Println(slice1)
	fmt.Println(slice2)

	//slice1和slice2的地址是一樣的
	fmt.Println("slice1地址:", (*reflect.SliceHeader)(unsafe.Pointer(&slice1)))
	fmt.Println("slice2地址:", (*reflect.SliceHeader)(unsafe.Pointer(&slice2)))
}

運行後輸出如下:

[1 2 3 4 5 6]
[1 2 3 4 5 6]
[1 100 3 4 5 6]
[1 100 3 4 5 6]
slice1地址: &{824633811232 6 6}
slice2地址: &{824633811232 6 6}

可以看出兩個對象的地址是一樣的。

要實現slice的深拷貝,就需要用到copy方法瞭,copy方法返回結果為一個int值,表示從原切片復制到目的切片的長度。在使用copy方法時,需要先初始化目的切片的長度:

  • 如果 dst 長度小於 src 的長度,則 拷貝src中的部分內容;
  • 如果大於,則全部拷貝過來,其餘的空間填充該類型的默認值;
  • 如果相等,剛好不多不少 copy 過來,所以,通常dst在初始化時即指定其為src的長度。

示例如下:

package main

import (
	"fmt"
	"reflect"
	"unsafe"
)

func main() {
	src := []int{1, 2, 3, 4, 5, 6}

	//輸出一下src的初度和值
	fmt.Println("src長度:", len(src), "src:", src)
	//輸出一下src地址
	fmt.Println("src地址:", (*reflect.SliceHeader)(unsafe.Pointer(&src)))

	dst1 := make([]int, 2)
	copy(dst1, src)
	fmt.Println("dst1長度:", len(dst1), "dst1:", dst1)
	fmt.Println("dst1地址:", (*reflect.SliceHeader)(unsafe.Pointer(&dst1)))

	dst2 := make([]int, len(src))
	copy(dst2, src)
	fmt.Println("dst2長度:", len(dst2), "dst2:", dst2)
	fmt.Println("dst2地址:", (*reflect.SliceHeader)(unsafe.Pointer(&dst2)))

	dst3 := make([]int, len(src)+2)
	copy(dst3, src)
	fmt.Println("dst3長度:", len(dst3), "dst3:", dst3)
	fmt.Println("dst3地址:", (*reflect.SliceHeader)(unsafe.Pointer(&dst3)))
}

輸出如下:

src長度: 6 src: [1 2 3 4 5 6]
src地址: &{824633811232 6 6}
dst1長度: 2 dst1: [1 2]
dst1地址: &{824633819808 2 2}
dst2長度: 6 dst2: [1 2 3 4 5 6]
dst2地址: &{824633811280 6 6}
dst3長度: 8 dst3: [1 2 3 4 5 6 0 0]
dst3地址: &{824633843904 8 8}

可以看出新的對象和原對象的地址都是不同的。

小結

深拷貝是創建一個新對象,完全復制原始對象及其所有嵌套的對象,因此新的對象是原始對象的獨立拷貝,之後的修改不會影響原始對象。淺拷貝則隻拷貝原始對象的數據結構的地址引用,因此新的對象和原始對象的引用指向相同的底層數據結構,對新對象的修改也會影響到原始對象。

到此這篇關於Golang中深拷貝與淺拷貝詳解的文章就介紹到這瞭,更多相關Golang 深拷貝與淺拷貝內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: