Go 语言中的两种 slice 表达式

bizky · 2018-08-24 16:45:54 · 1905 次点击 · 预计阅读时间 4 分钟 · 大约8小时之前 开始浏览    
这是一个创建于 2018-08-24 16:45:54 的文章,其中的信息可能已经有所发展或是发生改变。

在此之前,已经有许多关于 Golang 中 slice 的介绍,比如

本文只是关注于 slice 的表示方式,它们可以创建两种类型的值:

  • 截断的 string
  • 指向 array 或者 slice 的指针

Go 语言对 slice 有两种表示方式:简略表达式与完整表达式。

简略表达式

Slice 的简略表达式是:

Input[low:high]

其中,low 和 high 是 slice 的索引(index),其数值必须是整数,它们指定了输入操作数(Input)的哪些元素可以被放置在结果的 slice 中。输入操作数可以是 string,array,或者是指向 array 或 slice 的指针。结果 slice 的长度就是 high-low。如下例所示:

numbers := [10]int{0,1,2,3,4,5,6,7,8,9}
s := numbers[2:4:6]
fmt.Println(s) // [2, 3]
fmt.Println(cap(s)) // 4

将 slice 表达式应用到 array 的指针中,这是第一次取消对该指针的引用,然后以常规的方式应用 slice 表达式。将 slice 表达式应用于数组的指针,是先解引用该指针,然后按常规方式应用切片表达式的简写形式。

numbers := [5]int{1, 2, 3, 4, 5}
fmt.Println((&numbers)[1:3]) // [2, 3]

Slice 的索引 low 和 high 可以省略,low 的默认值是0,high 的默认值为 slice 的长度:

fmt.Println("foo"[:2]) // "fo"
fmt.Println("foo"[1:]) // "oo"
fmt.Println("foo"[:]) // "foo"

但是 slice 的索引不可以是以下几种类型的值:

  • low < 0 or high < 0
  • low <= high
  • high <= len(input)
fmt.Println("foo"[-1:])  // invalid slice index -1 (index must be non-negative)
//fmt.Println("foo"[:4]) // invalid slice index 4 (out of bounds for 3-byte string)
fmt.Println("foo"[2:2])  // ""(blank)
//fmt.Println("foo"[2:1]) // invalid slice index: 2 > 1

否则,即使一个超过 slice 范围的索引不能在编译的时候被检测到,那么在运行时就会发生一个 panic。

func low() int {
    return 4
}

func main() {
    fmt.Println("foo"[low():])
}
panic: runtime error: slice bounds out of range

goroutine 1 [running]:
panic(0x102280, 0x1040a018)
    /usr/local/go/src/runtime/panic.go:500 +0x720
main.main()
    /tmp/sandbox685025974/main.go:12 +0x120

完整表达式

这种方法可以控制结果 slice 的容量,但是只能用于 array 和指向 array 或 slice 的指针( string 不支持),在简略表达式中结果 slice 的容量是从索引low开始的最大可能容量( slice 的简略表达式):

numbers := [10]int{0,1,2,3,4,5,6,7,8,9}
s := numbers[1:4]
fmt.Println(s) // [1, 2, 3]
fmt.Println(cap(s)) // 9

对于一个 array 来说,cap(a) == len(a) 在上面的代码片段中,s 的容量是 9,因为这个 slice 从索引 1 开始,而底层 array 有 8 个元素(2 到 9)。完整的 slice 表达式允许修改这种默认行为,以如下代码为例:

numbers := [10]int{0,1,2,3,4,5,6,7,8,9}
s := numbers[1:4:5]
fmt.Println(s) // [1, 2, 3]
fmt.Println(cap(s)) // 4

完整的 slice 表达式具有以下的形式:

input[low:high:max]

索引 low 和索引 high 的含义和工作方式与简略表达式相同。唯一的区别是 max 将结果 slice 的容量设置为 max-low。

numbers := [10]int{0,1,2,3,4,5,6,7,8,9}
s := numbers[2:4:6]
fmt.Println(s) // [2, 3]
fmt.Println(cap(s)) // 4

当 slice 的输入操作数是一个 slice 时,结果 slice 的容量取决于输入操作数,而不是它的指向的底层 array:

numbers := [10]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
fmt.Println(cap(numbers)) // 10
s1 := numbers[1:4]
fmt.Println(s1) // [1, 2, 3]
fmt.Println(cap(s1)) // 9
s2 := numbers[1:4:5]
fmt.Println(s2) // [1, 2, 3]
fmt.Println(cap(s2)) // 4
s3 := s2[:]
fmt.Println(s3) // [1, 2, 3]
fmt.Println(cap(s3)) // 4

这个例子中 s3 的容量不能超过 s2 的容量 (4),即使它指向的 array 有 10 个元素,而且 s1,s2,s3 都是从 1 开始的。

当输入操作数是 slice 时,完整表达式的索引 high 不能超过其 cap(input)

numbers := [10]int{0,1,2,3,4,5,6,7,8,9}
s1 := numbers[0:1]
fmt.Println(s1) // [0]
fmt.Println(len(s1)) // 1
fmt.Println(cap(s1)) // 10
s2 := numbers[0:5]
fmt.Println(s2) // [0, 1, 2, 3, 4]
fmt.Println(cap(s1)) // 10

另外,对于它的 max 取值,有两个额外的规则

  • high <= max
  • max <= cap(input)
numbers := [10]int{0,1,2,3,4,5,6,7,8,9}
s1 := numbers[0:1]
s2 := numbers[0:5:11] // invalid slice index 11 (out of bounds for 10-element array)
fmt.Println(s1, s2)

在完整的 slice 表达式中低索引(low)是可以省略的

numbers := [10]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
s := numbers[:4:6]
fmt.Println(s) // [0, 1, 2, 3]
fmt.Println(cap(s)) // 6

然而,不能省略略高索引(high)

numbers := [10]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
s1 := numbers[:4:6]
s2 := s1[::5]
fmt.Println(s2)
fmt.Println(cap(s2))

否则,代码会在编译时报错"middle index required in 3-index slice"


via: https://medium.com/golangspec/slice-expressions-in-go-963368c20765

作者:Michał Łowicki  译者:bizky  校对:polaris1119

本文由 GCTT 原创编译,Go语言中文网 荣誉推出


有疑问加站长微信联系(非本文作者))

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

1905 次点击  
加入收藏 微博
被以下专栏收入,发现更多相似内容
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传