欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Golang中的深拷貝與淺拷貝使用

 更新時間:2023年04月03日 15:16:26   作者:JackMa_  
本文主要介紹了Golang中的深拷貝與淺拷貝使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

一、概念

1、深拷貝(Deep Copy)

拷貝的是數(shù)據(jù)本身,創(chuàng)造一個樣的新對象,新創(chuàng)建的對象與原對象不共享內(nèi)存,新創(chuàng)建的對象在內(nèi)存中開辟一個新的內(nèi)存地址,新對象值修改時不會影響原對象值。既然內(nèi)存地址不同,釋放內(nèi)存地址時,可分別釋放。

值類型的數(shù)據(jù),默認(rèn)全部都是深復(fù)制,Array、Int、String、Struct、Float,Bool。

2、淺拷貝(Shallow Copy)

拷貝的是數(shù)據(jù)地址,只復(fù)制指向的對象的指針,此時新對象和老對象指向的內(nèi)存地址是一樣的,新對象值修改時老對象也會變化。釋放內(nèi)存地址時,同時釋放內(nèi)存地址。

引用類型的數(shù)據(jù),默認(rèn)全部都是淺復(fù)制,Slice,Map。

二、本質(zhì)區(qū)別

是否真正獲取(復(fù)制)對象實(shí)體,而不是引用。

三、示例

淺拷貝

等號賦值

package main
 
import (
    "fmt"
    "reflect"
    "unsafe"
)
 
func main() {
    slice1 := []int{1,2,3,4,5}
    slice2 := slice1
    fmt.Println(slice1)
    fmt.Println(slice2)
    //同時改變兩個數(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é)果為一個 int 型值,表示 copy 從原切片src復(fù)制到目的切片的長度。

使用注意事項(xiàng):

切片 dst 需要先初始化長度

在使用copy將 src 完全 復(fù)制 到 dst 時,需要初始化目的切片dst的長度。 

1.如果 dst 長度小于 src 的長度,則 拷貝src中的部分內(nèi)容;
2.如果大于,則全部拷貝過來,其余的空間填充該類型的默認(rèn)值;
3.如果相等,剛好不多不少 copy 過來,所以,通常dst在初始化時即指定其為src的長度。 

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]

源切片中元素類型為引用類型時,拷貝的是引用

由于copy 函數(shù),拷貝的是切片中的元素,所以如果切片元素的類型是引用類型,那么 copy 的也將是個引用。

如下面例子,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 多維切片中的每一個切片類型的元素,那么就需要將每個切片元素進(jìn)行 初始化 并 拷貝。注意是兩步:

1.先初始化每維切片,
2.再拷貝。
正確拷貝一個多維數(shù)組的打開方式:

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])) // 注意初始化長度
        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和等號復(fù)制的區(qū)別

1.性能方面:copy復(fù)制會比等號復(fù)制慢。 2.復(fù)制方式:copy復(fù)制為值復(fù)制,改變原切片的值不會影響新切片。而等號復(fù)制為指針復(fù)制,改變原切片或新切片都會對另一個產(chǎn)生影響。

深拷貝

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

對于深拷貝,任何對象都會被完完整整的拷貝一份,拷貝對象與被拷貝對象不存在任何聯(lián)系,也就不會互相影響。

如果你需要拷貝的對象中沒有引用類型,那么對于Golang而言使用淺拷貝就可以了。

基于序列化和反序列化來實(shí)現(xiàn)對象的深度拷貝:

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)
}

參考:

go深拷貝和淺拷貝

golang copy 函數(shù)的使用 

到此這篇關(guān)于Golang中的深拷貝與淺拷貝使用的文章就介紹到這了,更多相關(guān)Golang 深拷貝與淺拷貝內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • go語言中os包的用法實(shí)戰(zhàn)大全

    go語言中os包的用法實(shí)戰(zhàn)大全

    Go在os中提供了文件的基本操作,包括通常意義的打開、創(chuàng)建、讀寫等操作,除此以外為了追求便捷以及性能上,Go還在io/ioutil以及bufio提供一些其他函數(shù)供開發(fā)者使用,這篇文章主要給大家介紹了關(guān)于go語言中os包用法的相關(guān)資料,需要的朋友可以參考下
    2024-02-02
  • Golang關(guān)鍵字defer的用法詳解

    Golang關(guān)鍵字defer的用法詳解

    defer是Go里面的一個關(guān)鍵字,用在方法或函數(shù)前面,作為方法或函數(shù)的延遲調(diào)用。這篇文章主要為大家介紹了defer的簡單使用,需要的可以參考一下
    2023-05-05
  • 詳解minio分布式文件存儲

    詳解minio分布式文件存儲

    MinIO 是一款基于 Go 語言的高性能、可擴(kuò)展、云原生支持、操作簡單、開源的分布式對象存儲產(chǎn)品,這篇文章主要介紹了minio分布式文件存儲,需要的朋友可以參考下
    2023-10-10
  • Go pprof內(nèi)存指標(biāo)含義備忘錄及案例分析

    Go pprof內(nèi)存指標(biāo)含義備忘錄及案例分析

    這篇文章主要介紹了Go pprof內(nèi)存指標(biāo)含義備忘錄問題,小編特此把問題及案例分享到腳本之家平臺供大家學(xué)習(xí),需要的朋友可以參考下
    2020-03-03
  • Gotify搭建你的消息推送系統(tǒng)

    Gotify搭建你的消息推送系統(tǒng)

    這篇文章主要介紹了Gotify搭建你的消息推送系統(tǒng),今天要分享的是 gotify,是一個用 go 編寫的消息服務(wù)端,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2024-01-01
  • 一文教你如何封裝安全的go

    一文教你如何封裝安全的go

    這篇文章主要給大家介紹了關(guān)于如何封裝安全go的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2022-02-02
  • Golang多線程爬蟲高效抓取大量數(shù)據(jù)的利器

    Golang多線程爬蟲高效抓取大量數(shù)據(jù)的利器

    Golang多線程爬蟲是一種高效抓取大量數(shù)據(jù)的利器。Golang語言天生支持并發(fā)和多線程,可以輕松實(shí)現(xiàn)多線程爬蟲的開發(fā)。通過使用Golang的協(xié)程和通道,可以實(shí)現(xiàn)爬蟲的高效并發(fā)抓取、數(shù)據(jù)處理和存儲
    2023-05-05
  • go mutex互斥鎖使用Lock和Unlock方法占有釋放資源

    go mutex互斥鎖使用Lock和Unlock方法占有釋放資源

    Go號稱是為了高并發(fā)而生的,在高并發(fā)場景下,勢必會涉及到對公共資源的競爭,當(dāng)對應(yīng)場景發(fā)生時,我們經(jīng)常會使用 mutex 的 Lock() 和 Unlock() 方法來占有或釋放資源,雖然調(diào)用簡單,但 mutex 的內(nèi)部卻涉及挺多的,本文來好好研究一下
    2023-09-09
  • Go 語言入門學(xué)習(xí)之正則表達(dá)式

    Go 語言入門學(xué)習(xí)之正則表達(dá)式

    這篇文章主要介紹了Go 語言入門學(xué)習(xí)之正則表達(dá)式,文章基于GO語言的相關(guān)資料展開詳細(xì)內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下
    2022-04-04
  • Go語言defer與return執(zhí)行的先后順序詳解

    Go語言defer與return執(zhí)行的先后順序詳解

    這篇文章主要為大家介紹了Go語言defer與return執(zhí)行的先后順序詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-12-12

最新評論