切片的长度是切片中元素的数量
切片的容量是从创建切片的索引开始的底层数组中元素的数量
切片是可索引的,并且可以由len()方法获取长度,切片提供了计算容量的方法cap(),可以测量切片最长可以达到多少。
切片实际的是获取数组的某一部分,len切片<=cap切片<=len数组
cap()的结果决定了切片截取的注意细节
var sTest01 []int
func sliceTest01() {
fmt.Printf("%T \n cap(sTest01) = %v \n", sTest01, cap(sTest01))
sTest01 = append(sTest01, 1, 2, 3, 4, 5, 6)
fmt.Printf("%T \n cap(sTest01) = %v \n", sTest01, cap(sTest01))
}
运行结果:
[]int
cap(sTest01) = 0
[]int
cap(sTest01) = 6
由此可以看到,在初始时,切片长度为0,在添加元素后,实际长度为6
证明数组是值类型,切片是引用类型的例子:
func sliceTest02() {
x := [...]int{1, 2, 3, 4, 5, 6}
y := []int{100, 200, 300, 400}
w := x
z := y
w[0] = 777
z[0] = 999
fmt.Println("x = ", x, "\nw = ", w)
fmt.Println("y = ", y, "\nz = ", z)
}
运行结果:
x = [1 2 3 4 5 6]
w = [777 2 3 4 5 6]
y = [999 200 300 400]
z = [999 200 300 400]
从运行结果可以看到,z的改变影响到了y的值,说明切片是引用类型。
slice没有自己的任何数据,它只是底层数组的一个引用,对slice所做的任何修改都将反映在底层数组中数组是值类型,而切片是引用类型
func sliceCap() {
arr := [...]string{"a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k"}
fmt.Println("cap(arr) = ", cap(arr), arr)
//截取数组,形成切片
s1 := arr[2:8]
fmt.Printf("%T \n", s1)
fmt.Println("cap(s1) = ", cap(s1), s1)
//截取数组,形成切片
s2 := arr[4:7]
fmt.Printf("%T \n", s2)
fmt.Println("cap(s2) = ", cap(s2), s2)
//截取数组,形成切片
s3 := s1[3:9]
fmt.Printf("%T \n", s3)
fmt.Println("cap(s3) = ", cap(s3), s3)
//截取数组,形成切片
s4 := s2[4:7]
fmt.Printf("%T \n", s4)
fmt.Println("cap(s4) = ", cap(s4), s4)
//证明切片是引用类型
s4[0] = "x"
fmt.Println(arr, s1, s2, s3, s4)
}
运行结果:
cap(arr) = 11 [a b c d e f g h i j k]
[]string
cap(s1) = 9 [c d e f g h]
[]string
cap(s2) = 7 [e f g]
[]string
cap(s3) = 6 [f g h i j k]
[]string
cap(s4) = 3 [i j k]
[a b c d e f g h x j k] [c d e f g h] [e f g] [f g h x j k] [x j k]
由结果我们可以看到,切片的截取只能生成切片,如果内容不够截取,会从底层数组给出后面的值。如果长度不够,会报错。
判断一个切片是否为空,如果直接使用nil来判断,是不准确的。
Golang允许len为0但是cap不为0,或者两者都为的0的切片,所以一般通过len获取切片长度来判断是否为空切片,而不是直接将切片和nil做直接的比较。