golang:为什么以下代码不会因“切片索引超出范围”错误而感到恐慌

时间:2017-01-07 16:02:59

标签: go slice

package main

import "fmt"

type Point struct {
        X int
        Y int
}

type Points struct {
    P []Point
}

func main() {
    data := Points{}
    for i := 0; i < 10; i++ {
        data.P = append(data.P, Point{
            X: i,
            Y: i*2,
        })
    }
    fmt.Printf("%+v\n", data.P[5:11]);
}

在运行上述程序时,打印出来:

[{X:5 Y:10} {X:6 Y:12} {X:7 Y:14} {X:8 Y:16} {X:9 Y:18} {X:9 Y:18} {X:0 Y: 0}]

为什么有{X:0,Y:0}似乎是自动生成的,因为切片的长度是10但是我试着得到5:11?

我在代码中发现了问题并使用“原始”切片进行测试,例如:

package main

import "fmt"

func main() {
    v := []int{0,1,2,3,4,5,6,7,8,9}
    fmt.Printf("%v\n", v[5:11])
}

这个简单的程序会产生错误(如预期的那样):

panic: runtime error: slice bounds out of range

goroutine 1 [running]:
panic(0x47a8e0, 0xc42000a130)
        /usr/local/go/src/runtime/panic.go:500 +0x1a1
main.main()
        /home/fxr/go/src/cmhwc/test2.go:7 +0x53
exit status 2

为什么第一个程序不会出现恐慌?

1 个答案:

答案 0 :(得分:2)

因为当你使用append()将元素添加到一个没有更多空间来添加新元素的切片时,它会创建一个新的元素,使其复制旧元素的容量加倍,以及你要求的新元素。添加并返回对该引用的引用。当你使用切片语法要求超过切片明显末端的元素时,长度实际应用于创建的基础数组&amp;追加回来。

正如@JimB在评论中建议的那样,你可以添加一个语句来打印追加循环中每一步的容量,看看是否发生了这种情况:

fmt.Println(cap(data.P))

应该产生类似的东西:

2
2
4
4
8
8
8
8
16
16