什么是切片

切片是 Golang 中比较特殊的数据结构,这种数据结构更便于使用和管理数据集合。简单的说,切片就是一种简化版的动态数组。因为动态数组的长度不固定,所以切片的长度自然也就不能是类型的组成部分了。切片是围绕动态数组的概念构建的,是对数组的抽象。数组虽然有适用的地方,但是数组的类型和操作都不够灵活,因此在 Go 代码中数组使用的并不是很多,而切片则使用的相当广泛,理解切片的原理和用法相当重要。

切片的内部结构

我们先来看看切片的结构定义,即 reflect.SliceHeader:

type SliceHeader struct {
    Data uintptr
    Len int
    Cap int
}

由切片的结构定义可知,切片的结构由三个信息组成:

  1. 指针,指向底层数组中切片指定的开始位置

  2. 长度,即切片的长度

  3. 最大长度,也就是切片开始位置到数组的最后位置的长度

下图给出了切片 x := [2, 3, 5, 7, 9, 11, 13, 15] 和 y := [1:3] 两个切片对应的内存结构。

切片的创建和初始化

让我们看看切片有哪些创建和初始化的方式:

var(
  a []int // nil 切片,和 nil 相等,一般用来表示一个不存在的切片
  b = []int{} // 空切片,和 nil 不相等,一般用来表示一个空的集合  
  c = []int{1, 2, 3, 4} // 有 4 个元素的切片,len 和 cap 都为 4
  d = c[:2] // 有 2 个元素的切片,len 为 2,cap 为 4
  e = c[0:2:cap(c)] // 有 2 个元素的切片,len 为 2,cap 为 4
  f = c[:0] // 有 0 个元素的切片,len 为 0,cap 为 4
  g = make([]int, 3) // 有 3 个元素的切片,len 和 cap 都为 3
  h = make([]int, 3, 5) // 有 3 个元素的切片,len 为 3,cap 为 5
  i = make([]int, 0, 5) // 有 0 个元素的切片,len 为 0,cap 为 5
)

使用 Golang 内置的函数 len() 可以查看切片中有效元素的长度,内置的函数 cap() 可以查看切片容量大小。

修改切片

切片没有自己的任何数据,它只是底层数组的一个表示。对切片所做的任何修改都将反映在底层数组中。

示例代码:

package main

import (  
    "fmt"
)

func main() {  
    darr := [...]int{57, 89, 90, 82, 100, 78, 67, 69, 59}
    dslice := darr[2:5]
    fmt.Println("array before",darr)
    for i := range dslice {
        dslice[i]++
    }
    fmt.Println("array after",darr) 
}

运行结果:

array before [57 89 90 82 100 78 67 69 59]  
array after [57 89 91 83 101 78 67 69 59] 

当多个片共享相同的底层数组时,每个元素所做的更改将在数组中反映出来。

示例代码:

package main

import (  
    "fmt"
)

func main() {  
    numa := [3]int{78, 79 ,80}
    nums1 := numa[:] //creates a slice which contains all elements of the array
    nums2 := numa[:]
    fmt.Println("array before change 1",numa)
    nums1[0] = 100
    fmt.Println("array after modification to slice nums1", numa)
    nums2[1] = 101
    fmt.Println("array after modification to slice nums2", numa)
}

运行结果:

array before change 1 [78 79 80]  
array after modification to slice nums1 [100 79 80]  
array after modification to slice nums2 [100 101 80]