切片和数组很类似,甚至你可以理解成数组的子集。但是`切片有一个数组所没有的特点,那就是切片的长度是可变的`。
严格地讲,切片有`容量(capacity)`和`长度(length)`两个属性。
首先我们来看一下切片的定义。切片有两种定义方式,一种是先声明一个变量是切片,然后使用内置函数make去初始化这个切片。另外一种是通过取数组切片来赋值。
1 package main 2 3 import ( 4 "fmt" 5 ) 6 7 func main() { 8 var x = make([]float64, 5) 9 fmt.Println("Capcity:", cap(x), "Length:", len(x))10 var y = make([]float64, 5, 10)11 fmt.Println("Capcity:", cap(y), "Length:", len(y))12 13 for i := 0; i < len(x); i++ {14 x[i] = float64(i)15 }16 fmt.Println(x)17 18 for i := 0; i < len(y); i++ {19 y[i] = float64(i)20 }21 fmt.Println(y)22 }
输出结果为
Capcity: 5 Length: 5
Capcity: 10 Length: 5 [0 1 2 3 4] [0 1 2 3 4]上面我们首先用make函数定义切片x,这个时候x的容量是5,长度也是5。然后使用make函数定义了切片y,这个时候y的容量是10,长度是5。然后我们再分别为切片x和y的元素赋值,最后输出。
所以使用make函数定义切片的时候,有`两种方式`,一种`只指定长度,这个时候切片的长度和容量是相同的`。另外一种是`同时指定切片长度和容量`。虽然切片的容量可以大于长度,但是`赋值的时候要注意最大的索引仍然是len(x)-1`。否则会报索引超出边界错误。
另外一种是通过数组切片赋值,采用`[low_index:high_index]`的方式获取数值切片,其中切片元素`包括low_index的元素`,但是`不包括high_index的元素`。
1 package main 2 3 import ( 4 "fmt" 5 ) 6 7 func main() { 8 var arr1 = [5]int{1, 2, 3, 4, 5} 9 var s1 = arr1[2:3]10 var s2 = arr1[:3]11 var s3 = arr1[2:]12 var s4 = arr1[:]13 fmt.Println(s1)14 fmt.Println(s2)15 fmt.Println(s3)16 fmt.Println(s4)17 }
输出结果为
[3]
[1 2 3] [3 4 5] [1 2 3 4 5] 在上面的例子中,我们还省略了low_index或high_index。如果省略了low_index,那么等价于从索引0开始;如果省略了high_index,则默认high_index等于len(arr1),即切片长度。这里为了体现切片的长度可以变化,我们看一下下面的例子:
1 package main 2 3 import ( 4 "fmt" 5 ) 6 7 func main() { 8 var arr1 = make([]int, 5, 10) 9 for i := 0; i < len(arr1); i++ {10 arr1[i] = i11 }12 fmt.Println(arr1)13 14 arr1 = append(arr1, 5, 6, 7, 8)15 fmt.Println("Capacity:", cap(arr1), "Length:", len(arr1))16 fmt.Println(arr1)17 }
输出结果为
[0 1 2 3 4]
Capacity: 10 Length: 9 [0 1 2 3 4 5 6 7 8]这里我们初始化arr1为容量10,长度为5的切片,然后为前面的5个元素赋值。然后输出结果。然后我们再使用Go内置方法append来为arr1追加四个元素,这个时候再看一下arr1的容量和长度以及切片元素,我们发现切片的长度确实变了。
另外我们再用`append`方法给arr1多追加几个元素,试图超过arr1原来定义的容量大小。
1 package main 2 3 import ( 4 "fmt" 5 ) 6 7 func main() { 8 var arr1 = make([]int, 5, 10) 9 for i := 0; i < len(arr1); i++ {10 arr1[i] = i11 }12 13 arr1 = append(arr1, 5, 6, 7, 8, 9, 10)14 fmt.Println("Capacity:", cap(arr1), "Length:", len(arr1))15 fmt.Println(arr1)16 }
输出结果为
Capacity: 20 Length: 11
[0 1 2 3 4 5 6 7 8 9 10]我们发现arr1的长度变为11,因为元素个数现在为11个。另外我们发现arr1的容量也变了,变为原来的两倍。这是因为`Go在默认的情况下,如果追加的元素超过了容量大小,Go会自动地重新为切片分配容量,容量大小为原来的两倍`。上面我们介绍了,可以`使用append函数给切片增加元素`,现在我们再来介绍一个`copy函数用来从一个切片拷贝元素到另一个切片`。
1 package main 2 3 import ( 4 "fmt" 5 ) 6 7 func main() { 8 slice1 := []int{1, 2, 3, 4, 5, 6} 9 slice2 := make([]int, 5, 10)10 copy(slice2, slice1)11 fmt.Println(slice1)12 fmt.Println(slice2)13 }
输出结果
[1 2 3 4 5 6]
[1 2 3 4 5]在上面的例子中,我们将slice1的元素拷贝到slice2,因为slice2的长度为5,所以最多拷贝5个元素。总结一下,数组和切片的区别就在于`[]`里面是否有数字或者`...`。因为数值长度是固定的,而切片是可变的。