Golang range slice 与range array 之间的区别
结构图:
为什么?
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]]
到此这篇关于Golang range slice 与range array 之间的区别的文章就介绍到这了,更多相关Golang range slice 与 range array 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
Golang 内存模型The Go Memory Model
这篇文章主要为大家介绍了Golang 内存模型The Go Memory Model实例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪2022-11-11
最新评论