詳解Go語言中for range的"坑"
前言
Go 中的for range組合可以和方便的實(shí)現(xiàn)對一個(gè)數(shù)組或切片進(jìn)行遍歷,但是在某些情況下使用for range時(shí)很可能就會被"坑",下面用一段代碼來模擬下:
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)
}
}
代碼解析:
- 創(chuàng)建一個(gè)int slice,變量名為arr1并初始化 1,2,3 作為切片的值。
- 創(chuàng)建一個(gè)*int slice,變量名為arr2。
- 通過for range遍歷arr1,然后獲取每一個(gè)元素的指針,賦值到對應(yīng)arr2中。
- 逐行打印arr2中每個(gè)元素的值。
從代碼上看,打印出來的結(jié)果應(yīng)該是
1
2
3
然而真正的結(jié)果是
3
3
3
原因
因?yàn)閒or range在遍歷值類型時(shí),其中的v變量是一個(gè)值的拷貝,當(dāng)使用&獲取指針時(shí),實(shí)際上是獲取到v這個(gè)臨時(shí)變量的指針,而v變量在for range中只會創(chuàng)建一次,之后循環(huán)中會被一直重復(fù)使用,所以在arr2賦值的時(shí)候其實(shí)都是v變量的指針,而&v最終會指向arr1最后一個(gè)元素的值拷貝。
來看看下面這個(gè)代碼,用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)
}
}
使用臨時(shí)變量
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甚至將其寫入了文檔的『常見錯(cuò)誤』部分:文檔
到此這篇關(guān)于詳解Go語言中for range的"坑"的文章就介紹到這了,更多相關(guān)Go語言for range內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
go語言channel實(shí)現(xiàn)多核并行化運(yùn)行的方法
這篇文章主要介紹了go語言channel實(shí)現(xiàn)多核并行化運(yùn)行的方法,實(shí)例分析了channel實(shí)現(xiàn)多核并行化運(yùn)行的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03
Go?net?http超時(shí)應(yīng)用場景全面詳解
HTTP是一個(gè)復(fù)雜的多階段協(xié)議,因此沒有一個(gè)一刀切的超時(shí)解決方案,在這篇文章中,我將分解您可能需要應(yīng)用超時(shí)的各個(gè)階段,并研究在服務(wù)器端和客戶端上執(zhí)行超時(shí)的不同方法2024-01-01
go?singleflight緩存雪崩源碼分析與應(yīng)用
這篇文章主要為大家介紹了go?singleflight緩存雪崩源碼分析與應(yīng)用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09
Go語言之io.ReadAtLeast函數(shù)的基本使用和原理解析
io.ReadAtLeast函數(shù)是Go語言標(biāo)準(zhǔn)庫提供的一個(gè)工具函數(shù),能夠從數(shù)據(jù)源讀取至少指定數(shù)量的字節(jié)數(shù)據(jù)到緩沖區(qū)中,這篇文章主要介紹了io.ReadAtLeast函數(shù)的相關(guān)知識,需要的朋友可以參考下2023-07-07
Golang中Map按照Value大小排序的方法實(shí)例
這篇文章主要給大家介紹了關(guān)于Golang中Map按照Value大小排序的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-03-03
Go初學(xué)者踩坑之go?mod?init與自定義包的使用
go?mod是go的一個(gè)模塊管理工具,用來代替?zhèn)鹘y(tǒng)的GOPATH方案,下面這篇文章主要給大家介紹了關(guān)于Go初學(xué)者踩坑之go?mod?init與自定義包的使用,需要的朋友可以參考下2022-10-10

