乐趣区

关于程序员:golang常见问题分享奇怪的指针

golang mistakes

定期更新总结应用 golang 过程中遇到的常见问题。

关注我,不迷路

2. 奇怪的指针

这个是应用 Go 语言常常会犯的谬误。当咱们遍历一个数组,如果用 range 获取变量保留到另一个数组或者 slice 时,会遇到令人困惑的问题。

func main() {slice := []int{1, 2, 3}
    var ptrSlice []*int
    for _, v := range slice {ptrSlice = append(ptrSlice, &v)
    }
    for _, v := range ptrSlice {fmt.Println(*v)
    }
}

输入:

3
3
3

起因剖析

对于 for range 语句的实现,从编译器源码 gofrontend/go/statements.cc/For_range_statement::lower_range_slice()办法中能够看到有如下正文:

 // The loop we generate:
  //   for_temp := range
  //   len_temp := len(for_temp)
  //   for index_temp = 0; index_temp < len_temp; index_temp++ {//           value_temp = for_temp[index_temp]
  //           index = index_temp
  //           value = value_temp
  //           original body
  //   }

go 代码示意如下:

sa := s
sv1 := 0
sn := len(sa)
v1 := sv1
v2 := nil
for ; sv1 < sn; sv1++ {tmp := sa[sv1]
  v1, v2 = sv1, tmp
}

通过下面代码能够发现,返回的永远都是 v2 的变量,也就是获取的是 v2 变量的地址,所以输入的后果都是3

正确做法:

应用 &slice[i] 代替&v,代码如下:

func main() {slice := []int{1, 2, 3}
    var ptrSlice []*int
    for i, _ := range slice {ptrSlice = append(ptrSlice, &slice[i])
    }
    for _, v := range ptrSlice {fmt.Println(*v)
    }
}

本文由 mdnice 多平台公布

退出移动版