在golang中,切片长度就是切片中元素的数量;切片容量是从创建切片的索引开始的底层数组中元素的数量,即从切片的第一个元素开始数,到其底层数组元素末尾的个数。切片的长度与容量是可以计算的,内置方法len()可获取长度,cap()可获取容量;在使用切片的过程中,如果使用append()让切片的长度大于了切片的容量,那么切片的容量会以双倍的形式自动扩容。
本教程操作环境:windows7系统、GO 1.18版本、Dell G3电脑。
Go 语言 的 切片 拥有长度和容量。
-
切片的长度是切片中元素的数量
-
切片的容量是从创建切片的索引开始的底层数组中元素的数量,即从它的第一个元素开始数,到其底层数组元素末尾的个数。
切片是可索引的,并且可以由len()方法获取长度,切片提供了计算容量的方法cap(),可以测量切片最长可以达到多少。我们在使用切片的过程中,如果使用 append 使切片的长度大于了切片的容量,那么切片的容量会以双倍的形式自动扩容。
切片实际的是获取数组的某一部分,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做直接的比较。
【