Golang range slice 和 range array 之间的不同

1,351 阅读2分钟

为什么?

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 语法中,valuefor 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 的切片 appenddata 后,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 这个数组的切片 appenddata 后,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]]

本文正在参加技术专题18期-聊聊Go语言框架