Go 中的重新切片:混乱与清晰
在 Go 中,切片是表示数据数组的一种强大而有效的方式。然而,理解它们的复杂性对于初学者来说可能具有挑战性。其中一个方面是重新切片切片的概念。
考虑以下代码:
package main import "fmt" func main() { a := make([]int, 5) printSlice("a", a) b := make([]int, 0, 5) printSlice("b", b) c := b[:2] printSlice("c", c) d := c[2:5] printSlice("d", d) } func printSlice(s string, x []int) { fmt.Printf("%s len=%d cap=%d %v\n", s, len(x), cap(x), x) }
程序创建四个切片:a、b、c 和 d。令人惊讶的结果是,作为 b 前两个元素的切片创建的切片 c 的容量为 5,而不是 2。
要理解这种行为,我们需要深入研究底层数组的概念。在 Go 中,切片不存储实际数据;相反,它们引用一个底层数组。创建切片时,它指向该底层数组的一个段。
在上面的示例中,当创建长度为 0 但容量为 5 的 b 时,它在底层数组中保留 5 个槽位。当从 b 创建切片 c 时,它成为 b 的前两个元素上的窗口。然而,底层数组以及 c 的容量仍然是 5,其中包括 b 中未使用的槽。
通过稍微修改代码来进一步说明重新切片的概念:
func main() { b := make([]int, 0, 5) c := b[:2] d := c[1:5] // this is equivalent to d := b[1:5] d[0] = 1 printSlice("c", c) printSlice("d", d) }
在这种情况下,修改 d 的值也会修改 c 的值,这表明 c 和 d 只是同一底层上的不同窗口array.
理解重新切片的概念对于在 Go 中有效地使用切片至关重要。它允许您创建动态调整以适应不同数据大小的切片,而无需复制或重新分配内存。
以上是Go Slices 中底层数组的重新切片行为如何?的详细内容。更多信息请关注PHP中文网其他相关文章!