Golang 數組與分片

與數組的區(qū)別

  • 數組是固定長度的,而分片確實可動態(tài)增長的,以定義為例:
// 定義數組, 一定要指定長度
var names [5]string

// 定義分片, 不需要指定長度
var names []string
  • 在函數調用時, 數組是值傳遞,而分片是引用傳遞

其實對于 golang 來講,函數調用的時候都是值傳遞,拷貝一個副本, 之所以表現為值傳遞和引用傳遞,在于一個拷貝的是數據值,另一個拷貝的是數據指針,兩個指針值指向的是同一個內存地址。

分片的實現

分片的底層數據還是使用的數組,它一共包含 3 個字段:

  1. 地址指針
  2. 長度
  3. 容量
// source 是一個分片, 大小為 4, 容量為: 5
source := make([]string, 4, 5)
fmt.Println(source, len(source), cap(source))
// 輸出: [   ] 4 5

// 注意這里不會進行內存分配, 因為 source 還有剩余空間可以新加數據
source = append(source, "1")
fmt.Println(source, len(source), cap(source))
// 輸出: [    1] 5 5

在使用 append 來為分片添加數據時, 是否有內存分配很重要

append 沒有內存分配時:

source := []string{"1", "2", "3", "4", "5"}

// 拷貝 source 的第二到第三個元素(不包括第三個)
// copied 的容量包括: 3, 4,5
copied := source[2:3]

fmt.Println("source-->", source)
fmt.Println("copied-->", copied)

fmt.Println("接下來修改 copied 的內容,看是否會對 source 產生影響...")

// 這里 append 內部不會創(chuàng)建一個新的底層數組,共有 source 的底層數組
// 因為 copied 的容量足夠新加一個元素
// 所以不會影響到 source 的內容
copied = append(copied, "mike")
fmt.Println("source-->", source)
fmt.Println("copied-->", copied)

輸出:

source--> [1 2 3 4 5]
copied--> [3]
接下來修改 copied 的內容,看是否會對 source 產生影響...
source--> [1 2 3 mike 5]
copied--> [3 mike]

append 有內存分配時:

source := []string{"1", "2", "3", "4", "5"}

// 拷貝 source 的第二到第三個元素(不包括第三個)
// copied 的容量包括: 3
copied := source[2:3:3]
// 此時 copied 會和 source 共享底層數組

fmt.Println("source-->", source)
fmt.Println("copied-->", copied)

fmt.Println("接下來修改 copied 的內容,看是否會對 source 產生影響...")

// 這里 append 內部會創(chuàng)建一個新的底層數組,不會共有 source 的底層數組
// 所以不會影響到 source 的內容
copied = append(copied, "mike")
fmt.Println("source-->", source)
fmt.Println("copied-->", copied)

輸出:

source--> [1 2 3 4 5]
copied--> [3]
接下來修改 copied 的內容,看是否會對 source 產生影響...
source--> [1 2 3 4 5]
copied--> [3 mike]
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容