首頁 >後端開發 >Golang >為什麼複製 Go 切片不會創造深層複製,如何避免意外突變?

為什麼複製 Go 切片不會創造深層複製,如何避免意外突變?

Patricia Arquette
Patricia Arquette原創
2024-11-27 18:38:14575瀏覽

Why Does Copying a Go Slice Not Create a Deep Copy, and How Can I Avoid Unintended Mutations?

了解Go 中的切片變異性

在Go 中,切片是一個動態大小的變量,用於儲存連續的元素序列。它是使用儲存長度和指向包含元素的底層數組的指標來實現的。

考慮以下程式碼片段:

func someFunc(A []int) int {
    ...
    tempA := A // copy the slice by value
    ...
    newArr = remove(tempA, i)
    ...
}

func remove(slice []int, s int) []int {
    return append(slice[:s], slice[s+1:]...)
}

在此程式碼中,append 函數用於修改tempA切片。然而,當 someFunc 函數傳回時,原始 A 切片也會被修改,儘管 tempA 是作為 A 的副本建立的。

為什麼會發生這種情況?

切片頭是與底層陣列分開的資料結構。複製切片時,僅複製標頭,而不複製底層數組。這意味著原始 A 切片和新建立的 tempA 切片都指向同一個底層陣列。

呼叫append 函數時,會建立一個新切片,並將 tempA 切片的元素複製到新切片。然而,A 切片仍然指向相同的底層數組,該數組現在包含修改後的元素。

建議

為了避免在修改 a 時原始切片發生意外突變複製時,您應該使用複製功能來建立切片的深層複製。複製函數建立一個新的底層數組並將元素複製到其中。

以下是如何使用複製函數的範例:

tempA := make([]int, len(A))
copy(tempA, A)

這將為以下物件建立一個新的底層陣列: tempA 並將 A 中的元素複製到其中。對 tempA 所做的任何修改都不會影響原始 A 切片。

以上是為什麼複製 Go 切片不會創造深層複製,如何避免意外突變?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn