Go語(yǔ)言中for循環(huán)的經(jīng)典案例分析
前言
for循環(huán)問(wèn)題,在面試中經(jīng)常都會(huì)被問(wèn)到,并且在實(shí)際業(yè)務(wù)項(xiàng)目中也經(jīng)常用到for循環(huán),要是沒(méi)用好,一不下心就掉坑。
下面會(huì)挑選幾個(gè)經(jīng)典的案例,一塊來(lái)探討下,看看如何避免掉坑,多積累積累采坑經(jīng)驗(yàn)。
案例一:for+傳值
先來(lái)到開(kāi)胃菜,熱熱身~
type student struct {
name string
age int
}
func main() {
m := make(map[string]student)
stus := []student{
{name: "張三", age: 18},
{name: "李四", age: 23},
{name: "王五", age: 26},
}
for _, stu := range stus {
m[stu.name] = stu
}
for k, v := range m {
fmt.Println(k, "=>", v.name)
}
}
不出意料,輸出結(jié)果為:
李四 => 李四
王五 => 王五
張三 => 張三
這題比較簡(jiǎn)單,就是簡(jiǎn)單的傳值操作,大家應(yīng)該都能答上來(lái)。下面加大難度,改為傳址操作
案例二:for+傳址
將案例一改為傳址操作
type student struct {
name string
age int
}
func main() {
m := make(map[string]*student)
stus := []student{
{name: "張三", age: 18},
{name: "李四", age: 23},
{name: "王五", age: 26},
}
for _, stu := range stus {
m[stu.name] = &stu
}
for k, v := range m {
fmt.Println(k, "=>", v.name)
}
}好好想想應(yīng)該輸出什么結(jié)果呢?還是跟案例一是一樣的結(jié)果嗎?難道會(huì)有坑?
不出意料,還是出了意外,輸出結(jié)果為:
張三 => 王五
李四 => 王五
王五 => 王五
為什么呢?
- 首先,關(guān)鍵點(diǎn)在于Go的for循環(huán),對(duì)
循環(huán)變量stu每次是循環(huán)并不是迭代(簡(jiǎn)單的說(shuō),就是對(duì)循環(huán)變量stu只會(huì)做一次聲明和內(nèi)存地址的分配,后面循環(huán)就是不斷更新值); - 所以,取址操作
&stu,其實(shí)都是取的同一個(gè)變量的地址,只是值被循環(huán)更新為最后一個(gè)元素的值; - 最終,輸出的
v.name,都是最后一個(gè)元素的name為王五。
解決方案:
在for循環(huán)中,做同名變量覆蓋stu:=stu(即重新聲明一個(gè)局部變量,做值拷貝,避免相互影響)
type student struct {
name string
age int
}
func main() {
m := make(map[string]*student)
stus := []student{
{name: "張三", age: 18},
{name: "李四", age: 23},
{name: "王五", age: 26},
}
for _, stu := range stus {
stu := stu //同名變量覆蓋
m[stu.name] = &stu
}
for k, v := range m {
fmt.Println(k, "=>", v.name)
}
}
輸出結(jié)果:
張三 => 張三
李四 => 李四
王五 => 王五
案例三:for+閉包
在for循環(huán)里,做閉包操作,也是很容易掉坑的??纯聪旅孑敵鍪裁矗?/p>
var prints []func()
for _, v := range []int{1, 2, 3} {
prints = append(prints, func() { fmt.Println(v) })
}
for _, print := range prints {
print()
}
一眼看過(guò)去,感覺(jué)是輸出1 2 3,但實(shí)際會(huì)輸出 3 3 3
為什么呢?
- 首先,在分析了案例二后,我們知道了Go的for循環(huán)對(duì)循環(huán)變量v,其實(shí)每次是循環(huán)并不是迭代;
- 然后,閉包=函數(shù)+引用環(huán)境,在同一個(gè)引用環(huán)境下,循環(huán)變量v的值會(huì)被不斷的覆蓋;
- 所以最終,在打印時(shí),輸出的v,都是最后一個(gè)值3。
解決方案:
和案例二解決方案一樣,是在for循環(huán)中,做同名變量覆蓋v:=v
var prints []func()
for _, v := range []int{1, 2, 3} {
v := v //同名變量覆蓋
prints = append(prints, func() { fmt.Println(v) })
}
for _, print := range prints {
print()
}
輸出結(jié)果:
1
2
3
案例四:for+goroutine
在for循環(huán)里,起goroutine協(xié)程,也是很迷惑很容易掉坑的??纯聪旅孑敵鍪裁??
var wg sync.WaitGroup
strs := []string{"1", "2", "3", "4", "5"}
for _, str := range strs {
wg.Add(1)
go func() {
defer wg.Done()
fmt.Println(str)
}()
}
wg.Wait()一眼看過(guò)去,感覺(jué)是會(huì)無(wú)序輸出1 2 3 4 5,但實(shí)際會(huì)輸出 5 5 5 5 5
為什么呢?
- 首先,要記得Go的for循環(huán)對(duì)循環(huán)變量str,其實(shí)每次是循環(huán)并不是迭代;
- 然后,main協(xié)程會(huì)和新起的協(xié)程做相互博弈,看誰(shuí)執(zhí)行更快,按這個(gè)案例執(zhí)行情況來(lái)看,main協(xié)程執(zhí)行速度明顯比新起的協(xié)程會(huì)更快,所以str被更新為最后一個(gè)元素值5(備注:并非絕對(duì));
- 最終,在新起的協(xié)程中,使用str時(shí)值都為5,作為結(jié)果去輸出;
- 拓展:如果在新起協(xié)程前,sleep個(gè)5s,輸出結(jié)果又會(huì)截然不同,感興趣的同學(xué)可以自行實(shí)驗(yàn)下,然后逐步深入地了解下GMP調(diào)度機(jī)制。
解決方案:
和前面兩個(gè)案例解決方案一樣,是在for循環(huán)中,做同名變量覆蓋str:=str
var wg sync.WaitGroup
strs := []string{"1", "2", "3", "4", "5"}
for _, str := range strs {
str := str //同名變量覆蓋
wg.Add(1)
go func() {
defer wg.Done()
fmt.Println(str)
}()
}
wg.Wait()
輸出結(jié)果:
5
4
2
1
3
注意是1~5無(wú)序輸出
總結(jié)
for循環(huán)中做傳址、閉包、goroutine相關(guān)操作,千萬(wàn)要注意,一不小心就會(huì)很容易掉坑。
到此這篇關(guān)于Go語(yǔ)言中for循環(huán)的經(jīng)典案例分析的文章就介紹到這了,更多相關(guān)Go for循環(huán)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- 深入解析Go語(yǔ)言中for循環(huán)的寫(xiě)法
- Go語(yǔ)言基礎(chǔ)for循環(huán)語(yǔ)句的用法及示例詳解
- 詳解如何在Go語(yǔ)言中循環(huán)數(shù)據(jù)結(jié)構(gòu)
- Go語(yǔ)言循環(huán)遍歷含有中文的字符串的方法小結(jié)
- 一文帶你熟悉Go語(yǔ)言中的for循環(huán)
- Go語(yǔ)言之fo循環(huán)與條件判斷
- 一文深入探索Go語(yǔ)言中的循環(huán)結(jié)構(gòu)
- Go語(yǔ)言中循環(huán)語(yǔ)句使用的示例詳解
- 一篇文章讓你學(xué)會(huì)Go語(yǔ)言循環(huán)語(yǔ)句
相關(guān)文章
go語(yǔ)言之給定英語(yǔ)文章統(tǒng)計(jì)單詞數(shù)量(go語(yǔ)言小練習(xí))
這篇文章給大家分享go語(yǔ)言小練習(xí)給定英語(yǔ)文章統(tǒng)計(jì)單詞數(shù)量,實(shí)現(xiàn)思路大概是利用go語(yǔ)言的map類(lèi)型,以每個(gè)單詞作為關(guān)鍵字存儲(chǔ)數(shù)量信息,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友參考下吧2020-01-01
golang 實(shí)現(xiàn)比特幣內(nèi)核之處理橢圓曲線中的天文數(shù)字
比特幣密碼學(xué)中涉及到的大數(shù)運(yùn)算超出常規(guī)整數(shù)范圍,需使用golang的big包進(jìn)行處理,通過(guò)使用big.Int類(lèi)型,能有效避免整數(shù)溢出,并保持邏輯正確性,測(cè)試展示了在不同質(zhì)數(shù)模下的運(yùn)算結(jié)果,驗(yàn)證了邏輯的準(zhǔn)確性,此外,探討了費(fèi)馬小定理在有限字段除法運(yùn)算中的應(yīng)用2024-11-11
Go基礎(chǔ)教程系列之import導(dǎo)入包(遠(yuǎn)程包)和變量初始化詳解
這篇文章主要介紹了Go基礎(chǔ)教程系列之import導(dǎo)包和初始化詳解,需要的朋友可以參考下2022-04-04
golang switch語(yǔ)句的靈活寫(xiě)法介紹
這篇文章主要介紹了golang switch語(yǔ)句的靈活寫(xiě)法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-05-05
Golang分布式注冊(cè)中心實(shí)現(xiàn)流程講解
這篇文章主要介紹了Golang分布式注冊(cè)中心實(shí)現(xiàn)流程,注冊(cè)中心可以用于服務(wù)發(fā)現(xiàn),服務(wù)注冊(cè),配置管理等方面,在分布式系統(tǒng)中,服務(wù)的發(fā)現(xiàn)和注冊(cè)是非常重要的組成部分,需要的朋友可以參考下2023-05-05
一文帶你了解Go語(yǔ)言fmt標(biāo)準(zhǔn)庫(kù)輸出函數(shù)的使用
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言中 fmt 標(biāo)準(zhǔn)庫(kù)輸出函數(shù)的使用,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起了解一下2022-12-12

