三索引切片主要是为了防止,在一个切片是建立在另一个切片之上时,扩充长度修改底层切片的内容。
正常语法slice[i:j]
,三索引语法slice[i:j:k]
可以简单理解在k没有指定情况下默认k=len(slice)
图示
func main() {
s1 := []int{1, 2, 3, 4, 5, 6, 7, 8}
s2 := s1[2:5]
fmt.Printf("s2 value: %v len: %d cap: %d\n", s2, len(s2), cap(s2))
s3 := s1[2:5:5]
fmt.Printf("s3 value: %v len: %d cap: %d\n", s3, len(s3), cap(s3))
}
输出
s2 value: [3 4 5] len: 3 cap: 6
s3 value: [3 4 5] len: 3 cap: 3
在不指定k情况下,上层切片会使用下层切片的长度,append
时会直覆盖底层切片的值,可能会引发意外情况。
func main() {
s1 := []int{1, 2, 3, 4, 5, 6, 7, 8}
s2 := s1[2:5]
s2 = append(s2, 0)
fmt.Printf("s1 value: %v len: %d cap: %d\n", s1, len(s1), cap(s1))
fmt.Printf("s2 value: %v len: %d cap: %d\n", s2, len(s2), cap(s2))
}
输出
s1 value: [1 2 3 4 5 0 7 8] len: 8 cap: 8
s2 value: [3 4 5 0] len: 4 cap: 6
使用k去限制切片,在发生扩容时会自动创建新的切片。
func main() {
s1 := []int{1, 2, 3, 4, 5, 6, 7, 8}
s3 := s1[2:5:5]
s3 = append(s3, 0)
fmt.Printf("s1 value: %v len: %d cap: %d\n", s1, len(s1), cap(s1))
fmt.Printf("s3 value: %v len: %d cap: %d\n", s3, len(s3), cap(s3))
}
输出
s1 value: [1 2 3 4 5 6 7 8] len: 8 cap: 8
s3 value: [3 4 5 0] len: 4 cap: 6