写个小demo来看看通过生成切片append扩容后发生了什么
package main
import (
"fmt"
)
var (
//实例化一个长度容量为5的slice
//从slice切出一个slice2
slice = make([]int, 5)
slice2 = slice[2:]
)
func main() {
printSlice()
//修改slice2第0个元素
slice2[0] = 1
printSlice()
//扩容slice2
slice2 = append(slice2, 2)
printSlice()
//修改slice2第0个元素
slice2[0] = 3
printSlice()
//修改slice第0个元素
slice[0] = 4
printSlice()
}
func printSlice() {
fmt.Printf("len=%d cap=%d %v\n", len(slice), cap(slice), slice)
fmt.Printf("len=%d cap=%d %v\n", len(slice2), cap(slice2), slice2)
}
打印信息如下
len=5 cap=5 [0 0 0 0 0]
len=3 cap=3 [0 0 0]
len=5 cap=5 [0 0 1 0 0]
len=3 cap=3 [1 0 0]
len=5 cap=5 [0 0 1 0 0]
len=4 cap=6 [1 0 0 2]
len=5 cap=5 [0 0 1 0 0]
len=4 cap=6 [3 0 0 2]
len=5 cap=5 [4 0 1 0 0]
len=4 cap=6 [3 0 0 2]
第一步我们创建了一个长度容量为5的slice,然后从slice的index为2的元素开始切到最后生成slice2,第二步我们将slice2的第0个元素改为1,打印slice和slice2,发现slice的第2个元素也变成了1。这也验证了slice和slice2共用了一个底层数组。
第二步我们对slice2进行扩容,增加一个元素为2,此时slice2的长度和容量分别为4和6。
第三步我们修改slice2的第0个元素为3,打印slice和slice2,发现slice的元素并未发生改变。
第四步修改了slice的第0个元素为4,打印后发现确实只有slice本身有变化,slice2没变化。
这验证了切片进行扩容的时候,会将引用的底层数组自身范围内的元素值,拷贝到扩容后的新的底层数组里。
有疑问加站长微信联系(非本文作者)