首頁 >後端開發 >Golang >當 Go 切片共享相同的底層數組時,為什麼追加到 Go 切片不會影響原始切片?

當 Go 切片共享相同的底層數組時,為什麼追加到 Go 切片不會影響原始切片?

Barbara Streisand
Barbara Streisand原創
2024-11-15 06:07:02266瀏覽

Why doesn't appending to a Go slice affect the original slice when they share the same underlying array?

了解切片上的追加行為

在 Go 中,切片是管理資料集合的便利方法。然而,它們的行為有時可能會令人困惑,特別是在執行附加元素等操作時。

考慮以下程式碼片段:

func main() {
    slice := make([]int, 10, 10)
    slice[0] = 0
    slice[1] = 1

    slice1 := slice
    slice1[0] = 10000
    fmt.Println(slice)

    slice1 = append(slice1, 100)
    slice1[0] = 20000

    fmt.Println(slice)
}

預期的輸出將是一個項目列表,其中第一個切片在第一次追加後更改為10000,然後在第二次追加後更改為20000。此外,我們希望 slice 中的項目也能反映這些更改,因為 slice 和 slice1 都指向相同的底層陣列。

但是,實際輸出令人驚訝:

[10000 1 0 0 0 0 0 0 0 0]
[10000 1 0 0 0 0 0 0 0 0]

第一個輸出如預期所示,顯示 slice1 成功將第一個元素的值更改為 10000。但是,第二個輸出顯示後續追加操作和對 slice1 的變更不會影響 slice。為什麼會這樣呢?

理解 Go 的值語意

理解這種行為的關鍵在於 Go 的值語意。在 Go 中,變數會以值傳遞,這表示當您將一個變數指派給另一個變數時,會產生該值的副本。這也適用於切片。

當您執行 slice1 := slice 時,您正在建立切片標頭的副本。切片頭包含長度、容量和指向底層數組的指標的資訊。但是,底層數組本身不會被複製。

追加行為

執行追加操作時,會建立一個新的切片頭並指派新的底層陣列。原始切片中的元素被複製到新數組中,並傳回新的切片頭。

在我們的範例中,當您執行 slice1 =append(slice1, 100) 時,將建立新的切片頭並為新陣列指派現有元素以及要附加的附加元素的空間。 slice1 中的元素被複製到新陣列中,並且新的切片頭被指派給 slice1。

重要的是,slice 仍然指向原始的底層數組,這就是為什麼對slice1 所做的更改不會反映在

結論

要理解切片上的追加行為,記住這一點至關重要切片按值傳遞。追加元素時,會建立新的切片頭和底層數組,而原始切片不受影響。

以上是當 Go 切片共享相同的底層數組時,為什麼追加到 Go 切片不會影響原始切片?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

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