Go | 三索引切片

Go | 三索引切片

1. 简介

三索引切片主要是为了防止,在一个切片是建立在另一个切片之上时,扩充长度修改底层切片的内容。

正常语法slice[i:j],三索引语法slice[i:j:k]

  • i:开始索引
  • j:结束索引(不包含自身)
  • k:容量(条件:j-i <= k <= len(slice))

可以简单理解在k没有指定情况下默认k=len(slice)

图示

Go | 三索引切片_第1张图片

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

2. 容易出错地方

在不指定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

3. 解决方法

使用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

4. 参考

  • Go 1.2 Release Notes

你可能感兴趣的:(Go,golang,go,切片,三索引)