您好,登錄后才能下訂單哦!
前言
Go 語言中的slice類型可以理解為是數組array類型的描述符,包含了三個因素:
因此當我們定義一個切片變量,s := make([]int, 5, 10),即為指向了一個最大長度為10的底層數組,目前切片s使用到的長度為5。
在使用切片的時候,有幾個注意事項,下面來一起看看吧。
使用append
先看一個例子:
// 創建一個整型切片 // 其長度和容量都是 5 個元素 slice := []int{10, 20, 30, 40, 50} // 創建一個新切片 // 其長度為 2 個元素,容量為 4 個元素 newSlice := slice[1:3] // 使用原有的容量來分配一個新元素 // 將新元素賦值為 60,會改變底層數組中的元素 newSlice = append(newSlice, 60) fmt.Println(slice, newSlice)
輸出:
[10 20 30 60 50] [20 30 60]
下圖可以非常形象的說明上述代碼的運行原理:
僅做一點點小的改變,結果就不一樣了:
// 創建一個整型切片 // 其長度和容量都是 5 個元素 slice := []int{10, 20, 30, 40, 50} // 創建一個新切片 // 其長度與容量相同 newSlice := slice[1:3:3] // 注意這里 // 使用原有的容量來分配一個新元素 // 將新元素賦值為 60,會改變底層數組中的元素 newSlice = append(newSlice, 60) // newSlice 的底層數組已經不是 slice 了,這個改變不會影響 slice newSlice[0] = 0 fmt.Println(slice, newSlice, cap(newSlice))
以上代碼會輸出:
[10 20 30 40 50] [0 30 60] 4
原因在于:當往 newSlice 中新增元素的時候,由于其容量不夠,newSlice 會擁有一個全新的底層數組,其容量是原來的兩倍(Go 會自動完成這個操作,一旦元素個數超過 1000,增長因子會設為 1.25)
使用 range 遍歷 slice
在使用 range 遍歷 slice 的時候,range 會創建每個元素的副本,看看這個例子:
slice := []int{10, 20, 30, 40} // 迭代每個元素,并顯示值和地址 for index, value := range slice { fmt.Printf("Value: %d Value-Addr: %X ElemAddr: %X\n", value, &value, &slice[index]) }
輸出:
Value: 10 Value-Addr: C420014060 ElemAddr: C420018080 Value: 20 Value-Addr: C420014060 ElemAddr: C420018088 Value: 30 Value-Addr: C420014060 ElemAddr: C420018090 Value: 40 Value-Addr: C420014060 ElemAddr: C420018098
可以看到 Value-Addr 跟 ElemAddr 的地址是不同的,印證了上面的說法。而每次迭代的變量的地址是相同的,說明迭代過程復用了這個變量,也是一種防止內存浪費的做法。
多維切片
創建一個多維切片:
// 創建一個整型切片的切片 slice := [][]int{{10}, {100, 200}}
其結構可以用下圖來表示:
其中第一維可以看成長度為 2,容量為 2 的保存了切片類型的切片,第二維則是整形切片。
其他規則則同處理一維切片一樣了,比如:
// 為第一個切片追加值為 20 的元素 slice[0] = append(slice[0], 20)
上述操作可以用下圖來表示:
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對億速云的支持。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。