目录
- 为什么?
- 理解
- case rangeSlice
- case rangeArray
- 测试代码
结构图:

为什么?
var data [][]int
for _, rangeSlice := range [][]int{{1}, {2}, {3}} {
data = append(data, rangeSlice[:])
}
fmt.Printf("%v", data) // 输出 [[1] [2] [3]]
var data [][]int
for _, rangeArray := range [][1]int{{1}, {2}, {3}} {
data = append(data, rangeArray[:])
}
fmt.Printf("%v", data) // 输出 [[3] [3] [3]]
理解
for key, value := range container{
// loop
}
在 for range
语法中,value
是 for range
循环返回元素的值的拷贝。
case rangeSlice
var data [][]int
for _, rangeSlice := range [][]int{{1}, {2}, {3}} {
data = append(data, rangeSlice[:])
}
fmt.Printf("%v", data) // 输出 [[1] [2] [3]]
代码中,rangeSlice
是切片 []int{1}
or []int{2}
or []int{3}
的一个拷贝,底层是数组 [1]int{1}
or [1]int{2}
or [1]int{3}
,所以在 rangeSlice
的切片 append
到 data
后,data
的元素也是这几个底层数组的切片,所以上述代码片段的输出为 [[1] [2] [3]]
。
case rangeArray
var data [][]int
for _, rangeArray := range [][1]int{{1}, {2}, {3}} {
data = append(data, rangeArray[:])
}
fmt.Printf("%v", data) // 输出 [[3] [3] [3]]
代码中,rangeArray
是数组 [1]int{1}
or [1]int{2}
or [1]int{3}
的一个拷贝,在每次循环中,rangeArray
用的都是同一个数组内存空间,所以在 rangeArray
这个数组的切片 append
到 data
后,data
的元素都是指向同一底层数组的切片,在循环的最后一轮 rangeArray
为 [1]int{3}
,所以上述代码片段的输出为 [[3] [3] [3]]
。
测试代码
package main
import "fmt"
func rangeSlice() {
source := [][]int{{1}, {2}, {3}}
var data [][]int
for idx, rangeSlice := range source {
fmt.Printf("%T %p %T %p", rangeSlice, rangeSlice, source[idx], source[idx])
fmt.Printf(" append slice %p\n", rangeSlice[:])
data = append(data, rangeSlice[:])
}
fmt.Printf("data: %v\n", data)
}
func rangeArray() {
source := [][1]int{{1}, {2}, {3}}
var data [][]int
for idx, rangeArray := range source {
fmt.Printf("%T %p %T %p", rangeArray, &rangeArray, source[idx], &source[idx])
fmt.Printf(" append slice %p\n", rangeArray[:])
data = append(data, rangeArray[:])
}
fmt.Printf("data: %v\n", data)
}
func main() {
rangeSlice()
rangeArray()
}
// 输出:
// []int 0xc00001a0a8 []int 0xc00001a0a8 append slice 0xc00001a0a8
// []int 0xc00001a0b0 []int 0xc00001a0b0 append slice 0xc00001a0b0
// []int 0xc00001a0b8 []int 0xc00001a0b8 append slice 0xc00001a0b8
// data: [[1] [2] [3]]
// [1]int 0xc00001a0e0 [1]int 0xc0000160f0 append slice 0xc00001a0e0
// [1]int 0xc00001a0e0 [1]int 0xc0000160f8 append slice 0xc00001a0e0
// [1]int 0xc00001a0e0 [1]int 0xc000016100 append slice 0xc00001a0e0
// data: [[3] [3] [3]]