Go 中的for range
組合能夠和方便的實現對一個數組或切片進行遍歷,可是在某些狀況下使用for range
時極可能就會被"坑"
,下面用一段代碼來模擬下:前端
func main() { arr1 := []int{1, 2, 3} arr2 := make([]*int, len(arr1)) for i, v := range arr1 { arr2[i] = &v } for _, v := range arr2 { fmt.Println(*v) } }
代碼解析:git
int slice
,變量名爲arr1
並初始化 1,2,3 做爲切片的值。*int slice
,變量名爲arr2
。for range
遍歷arr1
,而後獲取每個元素的指針,賦值到對應arr2
中。arr2
中每一個元素的值。從代碼上看,打印出來的結果應該是github
1 2 3
然而真正的結果是golang
3 3 3
由於for range
在遍歷值類型
時,其中的v
變量是一個值
的拷貝,當使用&
獲取指針時,其實是獲取到v
這個臨時變量的指針,而v
變量在for range
中只會建立一次,以後循環中會被一直重複使用,因此在arr2
賦值的時候其實都是v
變量的指針,而&v
最終會指向arr1
最後一個元素的值拷貝。docker
來看看下面這個代碼,用for i
來模擬for range
,這樣更易於理解:數組
func main() { arr1 := []int{1, 2, 3} arr2 := make([]*int, len(arr1)) var v int for i:=0;i<len(arr1);i++ { v = arr1[i] arr2[i] = &v } for _, v := range arr2 { fmt.Println(*v) } }
func main() { arr1 := []int{1, 2, 3} arr2 := make([]*int, len(arr1)) for i := range arr1 { arr2[i] = &arr1[i] } for _, v := range arr2 { fmt.Println(*v) } }
func main() { arr1 := []int{1, 2, 3} arr2 := make([]*int, len(arr1)) for i, v := range arr1 { t := v arr2[i] = &t } for _, v := range arr2 { fmt.Println(*v) } }
func main() { arr1 := []int{1, 2, 3} arr2 := make([]*int, len(arr1)) for i, v := range arr1 { func(v int){ arr2[i] = &v }(v) } for _, v := range arr2 { fmt.Println(*v) } }
因爲這一問題過於廣泛,Golang甚至將其寫入了文檔的『常見錯誤』部分:文檔閉包
我是 MonkeyWie,歡迎掃碼👇👇關注!不按期在公衆號中分享JAVA
、Golang
、前端
、docker
、k8s
等乾貨知識。