GoLang学习笔记(二十一)切片的长度和容量

切片的长度是切片中元素的数量
切片的容量是从创建切片的索引开始的底层数组中元素的数量
切片是可索引的,并且可以由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做直接的比较。

你可能感兴趣的:(GoLang)