go的切片是有指针、长度和容量三部分组成的。它本质是对底层数组的引用,本身不存储数据;也就是说一个切片的声明必定会对应一个底层数组。
示例:
//示例1:
//声明一个数组a
a:=[5]int{1,2,3,4}
//a[:]等价于a[0:4]创建一个数组a 的切片s, 数组a就是 切片s的底层数组
s := a[:]
//示例2:
//会先分配一个数组作为切片s的底层数组,然后返回一个引用该数组的切片
s:=[]int{1,2,3,4}
//nil切片长度为0,容量为0,没有底层数组
var s1 []int
切片的长度和容量
长度 为 切片引用的元素数量 ,使用 len() 获得长度;容量为底层数组的长度 与切片所引用的起始元素对应的下标数值 之 差。
声明一个长度为5的数组如下:
a := [5]int {1,2,3,4}
创建数组a的一个切片
s := a[1:3],指向数组a中下标为1和2的两个元素。s的长度为len(s) = 2; s的容量为 cap(s) = 4 即 len(a) - 1。
切片自动扩容时,会创建和指向新的底层数组,新的底层数组独立与原底层数组。
s := a[:] 声明一个数组a的切片,追加一个元素s = s.append(5) ,超出了s的最大容量,此时切片会自动追加容量,追加操作导致分配了新的底层数组,并指向这个新的数组值为 []int{1,2,3,4,0,5}