您好,登錄后才能下訂單哦!
這篇文章主要介紹“Go語言中的for循環(huán)使用實(shí)例分析”的相關(guān)知識,小編通過實(shí)際案例向大家展示操作過程,操作方法簡單快捷,實(shí)用性強(qiáng),希望這篇“Go語言中的for循環(huán)使用實(shí)例分析”文章能幫助大家解決問題。
問題
案例一:取地址符
在 Go 語言中,我們寫 for 語句時(shí)有時(shí)會出現(xiàn)運(yùn)行和猜想的結(jié)果不一致。例如以下第一個(gè)案例的代碼:
var all []*Itemfor _, item := range items {
all = append(all, &item)
}
這段代碼有問題嗎?變量 all 內(nèi)的 item 變量,存儲進(jìn)去的是什么? 是每次循環(huán)的 item 值嗎?
實(shí)際上在 for 循環(huán)時(shí),每次存入變量 all 的都是相同的 item,也就是最后一個(gè)循環(huán)的 item 值。
這是 Go 面試?yán)锝?jīng)常出現(xiàn)的題目,結(jié)合 goroutine 更風(fēng)騷,畢竟還會存在亂序輸出等問題。
如果你想解決這個(gè)問題,就需要把程序改寫成如下:
var all []*Itemfor _, item := range items {
item := item
all = append(all, &item)
}
要重新聲明一個(gè) item 變量把 for 循環(huán)的 item 變量給存儲下來再追加進(jìn)去。
案例二:閉包函數(shù)
接下來是第二個(gè)案例的代碼:
var prints []func()for _, v := range []int{1, 2, 3} {
prints = append(prints, func() { fmt.Println(v) })
}for _, print := range prints { print()
}
這段程序的輸出結(jié)果是什么?沒有 & 取地址符,是輸出 1,2,3 嗎?
輸出結(jié)果是 3,3,3。這又是為什么?
問題的重點(diǎn)之一,關(guān)注到閉包函數(shù),實(shí)際上所有閉包都打印的是相同的 v。輸出 3,是因?yàn)樵?for 循環(huán)結(jié)束后,最后 v 的值被設(shè)置為了 3,僅此而已。
如果想要達(dá)到預(yù)期的效果,依然是使用萬能的再賦值。改寫后的代碼如下:
for _, v := range []int{1, 2, 3} {
v := v
prints = append(prints, func() { fmt.Println(v) })
}
增加 v := v
語句,程序輸出結(jié)果為 1,2,3。
仔細(xì)翻翻你寫過的 Go 工程,是不是都很熟悉?就這改造方法,贏了。
尤其是配合上 Goroutine 的寫法,很多同學(xué)會更容易在此翻車。
解決方案
修復(fù)思路
實(shí)際上 Go 核心團(tuán)隊(duì)在內(nèi)部和社區(qū)已經(jīng)討論過許久,希望重新定義 for 循環(huán)的語法。要達(dá)到的目的是:使循環(huán)變量每次迭代而不是每次循環(huán)。
解決的辦法是:在每個(gè)迭代變量 x 的每個(gè)循環(huán)體開頭,加一個(gè)隱式的再賦值,也就是 x := x
,就能夠解決上述程序中所隱含的坑。和我們現(xiàn)在做的一樣,只不過我們是自己手動加的,Go 團(tuán)隊(duì)做的是希望在編譯器內(nèi)隱式處理。
讓用戶自己決定
比較尷尬的是 Go 團(tuán)隊(duì)在 Proposal: Go 2 transition 中禁止重新定義語言,所以 rsc 不能直接這么干。
因此將會由用戶自己決定控制這個(gè) “破壞”,方式將會是根據(jù)每個(gè)包的 go.mod 文件中的 go 行更改語義。
如果我們是在 Go1.30 對本文討論的 for 循環(huán)改為迭代,那么在 go.mod 文件中的 go 版本聲明是將是一個(gè)關(guān)鍵。
如下圖示:
Go 1.30 或更高版本將會每次迭代變量,而早期 Go 版本的將每次循環(huán)變量。
如此一來上述提到的 for 循環(huán)問題都會在一定范圍內(nèi)被解決。
關(guān)于“Go語言中的for循環(huán)使用實(shí)例分析”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識,可以關(guān)注億速云行業(yè)資訊頻道,小編每天都會為大家更新不同的知識點(diǎn)。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。