上一篇 ?詳解go語言的array和slice 【一】已經講解過,array和slice的一些基本用法,使用array和slice時需要注意的地方,特別是slice需要注意的地方比較多。上一篇的最后講解到創建新的slice時使用第三個索引來限制slice的容量,在操作新slice時,如果新slice的容量大于長度時,添加新元素依然后使源的相應元素改變。這一篇里我會講解到如何避免這些問題,以及迭代、和做為方法參數方面的知識點。
slice的長度和容量設置為同一個值
如果在創建新的slice時我們把他的長度和容量的值設置為樣的值,那么在append新元素時,底層會創建一個新的array并把之前的值復制過去。這樣就不會影響之前共同的底層array了。
// 創建一個容量和長度均為6的sliceslice1 := []int{5, 23, 10, 2, 61, 33}// 對slices1進行切片,長度為2容量為3slice2 := slice1[1:3:3]fmt.Println("cap", cap(slice2))fmt.Println("slice2", slice2)//修改一個共同指向的元素//兩個slice的值都會修改slice2[0] = 11111fmt.Println("slice1", slice1)fmt.Println("slice2", slice2)// 增加一個元素slice2 = append(slice2, 55555)fmt.Println("slice1: ", slice1)fmt.Println("slice2: ", slice2)
輸出結果
cap 2 slice2 [23 10] slice1 [5 11111 10 2 61 33] slice2 [11111 10] slice1: [5 11111 10 2 61 33] slice2: [11111 10 55555]
代碼中的長度和容量是一樣的,長度和容量的計算公式看我的上一篇博客。增加一個元素后,原來共同指向的底層數據是沒有變的。因為slice2的底層array被重新賦值了。
迭代slice
go語言內置一個關鍵字range用于迭代集合,當然他也可以迭代slice,也可以使用? _? 來忽略我們不關心的元素,但是如果只關心index則不需這么寫 for index,_ := range slice1。下在給出完整代碼
// 創建一個容量和長度均為6的sliceslice1 := []int{5, 23, 10, 2, 61, 33}for index, value := range slice1 {fmt.Println("index: ", index, " value: ", value)}// 可以忽略我們不關心的元素// 只關心valuefor _, value := range slice1 {fmt.Println("value ", value)}// 只關心index, 可以不用 _for index := range slice1 {fmt.Println("index: ", index)}
?
? 需要注意的是rang 迭代的value值并是一個復本,我們可以對比一下迭代的value和原slice內相應index下value的地址:
// 創建一個容量和長度均為6的sliceslice1 := []int{5, 23, 10, 2, 61, 33}for index, value := range slice1 {fmt.Println("index: ", index, " value address : ", &value, " slice1 value address", &slice1[index])}
輸出結果
index: 0 value address : 0xc04204e088 slice1 value address 0xc04206a030 index: 1 value address : 0xc04204e088 slice1 value address 0xc04206a038 index: 2 value address : 0xc04204e088 slice1 value address 0xc04206a040 index: 3 value address : 0xc04204e088 slice1 value address 0xc04206a048 index: 4 value address : 0xc04204e088 slice1 value address 0xc04206a050 index: 5 value address : 0xc04204e088 slice1 value address 0xc04206a058
slice1中value的地址是不斷變化的。而迭代的value值的地址沒有變化,這是因為value是一個變量,為次迭代的時候賦不同的值。我們把代碼寫成下面這樣,你就清楚了
var index, value intfor index, value = range slice1 {fmt.Println("index: ", index, &index, " value address : ", &value, " slice1 value address", &slice1[index])}
? 除了使用rang 也可以使用傳統的for循環來做迭代
slice1 := []int{5, 23, 10, 2, 61, 33}for i, len := 1, len(slice1); i < len; i++ {fmt.Println("index: ", i, " value:", slice1[i])}
?
?slice作為方法參數
? 由于slice的特殊結構,有一個指針指向一個數組
s := make([]int, 2, 5)fmt.Println("len: ", len(s))fmt.Println("cap: ", cap(s))s = append(s, 2)s[0] = 12
?
所以,slice做為方法的參數傳遞時,只會復制slice本身而不會復制slice底層的array.如果我們創建一個int類型有100萬長度的slice ,把他傳遞給一個方法時,只需要復制24個字節就夠了。指針需要8個,長度和容量都是8個。
const size int = 1000 * 1000func main() {slice0 := make([]int, size)fmt.Println("slice0 len: ", len(slice0), " cap :", cap(slice0))doSomeThing(slice0) }func doSomeThing(s []int) {fmt.Println(len(s)) }
?
?