golang遍歷時(shí)修改被遍歷對(duì)象的示例詳解
前言
很多時(shí)候需要將遍歷對(duì)象中去掉某些元素,或者往遍歷對(duì)象中添加元素,這時(shí)候就需要小心操作了。
對(duì)于go語(yǔ)言中的一些注意事項(xiàng)我做了總結(jié)和示例,留下點(diǎn)筆記。
遍歷切片
1.遍歷切片時(shí)去掉元素,錯(cuò)誤示例:
func main() {
arr := []int{1, 2, 3, 4}
for i := range arr {
if arr[i] == 3 {
arr = append(arr[:i], arr[i+1:]...)
}
}
fmt.Println(arr)
}
最終報(bào)錯(cuò)panic: runtime error: index out of range [3] with length 3,因?yàn)閞ange在迭代時(shí)已經(jīng)確定i的范圍為[0,len(arr))的左閉右開(kāi)的區(qū)間。
但是當(dāng)滿足arr[i] == 3時(shí)對(duì)arr進(jìn)行了修改,縮短了arr的長(zhǎng)度,此時(shí)len(arr)=3,最大下標(biāo)為2,因此當(dāng)執(zhí)行arr[3]時(shí)會(huì)報(bào)錯(cuò)。
2.遍歷切片時(shí)去掉元素,不會(huì)報(bào)錯(cuò),但不建議的寫(xiě)法:
func main() {
arr := []int{1, 2, 3, 4}
for i, v := range arr {
fmt.Println(i, v)
if v == 3 {
arr = append(arr[:i], arr[i+1:]...)
// arr = []int{4, 5, 6, 7} // 可以將上一行代碼替換,看最終遍歷的i,v情況
}
}
fmt.Println(arr)
}
還是回到range的用法,當(dāng)執(zhí)行for循環(huán)時(shí)就已經(jīng)確定(i,v)的遍歷元素值,及時(shí)循環(huán)過(guò)程中修改arr,也不會(huì)改變for要遍歷的(i,v)值。
可以將上面代碼修改一下,看下在循環(huán)中改變arr值時(shí),后面遍歷的(i,v)是不會(huì)隨著arr的改變而改變的。
3.遍歷切片時(shí)去掉元素,建議寫(xiě)法:
func main() {
arr := []int{1, 2, 3, 4}
for i := 0; i < len(arr); i++ {
fmt.Println(i, arr[i])
if arr[i] == 3 {
arr = append(arr[:i], arr[i+1:]...)
i--
}
}
fmt.Println(arr)
}
該方案只修改i的值,在刪除元素時(shí)進(jìn)行i--,可以確保遍歷arr沒(méi)有問(wèn)題,而且每次通過(guò)arr[i]獲取切片值不存在問(wèn)題。
當(dāng)然用該方式也可以在遍歷時(shí)添加元素,只要i也對(duì)應(yīng)變化就沒(méi)問(wèn)題。
遍歷map
1.遍歷map時(shí)去掉元素,可參考官方示例,可看下官方描述,下面這種方案是安全的。
for key := range m {
if key.expired() {
delete(m, key)
}
}
2.清空map所有元素,如下第一種是省事的寫(xiě)法,第二種不會(huì)產(chǎn)生新的對(duì)象,用哪種看個(gè)人喜好吧。
m = make(map[int]int) // 可以產(chǎn)生一個(gè)新對(duì)象,舊對(duì)象等著被垃圾回收
for k := range m {
delete(m,k) // 循環(huán)遍歷并刪除map所有元素,好處是map緩存還在,下次添加時(shí)可直接使用緩存
}
總結(jié)
關(guān)于切片遍歷時(shí)進(jìn)行操作需要注意一些坑。
關(guān)于map遍歷時(shí)進(jìn)行操作相對(duì)坑少點(diǎn),不過(guò)遍歷map需要修改元素時(shí),map的value要為指針類(lèi)型,這點(diǎn)得謹(jǐn)記。
到此這篇關(guān)于golang遍歷時(shí)修改被遍歷對(duì)象的文章就介紹到這了,更多相關(guān)golang遍歷對(duì)象內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
golang sync.Pool 指針數(shù)據(jù)覆蓋問(wèn)題解決
本文主要介紹了使用sync.Pool時(shí)遇到指針數(shù)據(jù)覆蓋的問(wèn)題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2025-03-03
golang?gorm學(xué)習(xí)之如何指定數(shù)據(jù)表
在sql中首先要指定是從哪張表中查詢,所以這篇文章小編就來(lái)帶大家一起看一下gorm是如何根據(jù)model來(lái)自動(dòng)解析表名的,感興趣的小伙伴可以了解下2023-08-08
詳解golang RWMutex讀寫(xiě)互斥鎖源碼分析
這篇文章主要介紹了詳解golang RWMutex讀寫(xiě)互斥鎖源碼分析,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-06-06
Golang 負(fù)載均衡算法實(shí)現(xiàn)示例
在Go語(yǔ)言中,負(fù)載均衡算法通常由代理、反向代理或者應(yīng)用層負(fù)載均衡器來(lái)實(shí)現(xiàn),在這些實(shí)現(xiàn)中,有一些經(jīng)典的負(fù)載均衡算法,跟隨本文來(lái)一一探究2024-01-01
詳解golang各種類(lèi)型是如何進(jìn)行比較的
在日常開(kāi)發(fā)中,比較操作是最常用的基本操作之一,可以用來(lái)判斷變量之間是否相等或者對(duì)應(yīng)的大小關(guān)系,比較操作對(duì)于排序、查找和集合數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)至關(guān)重要,本文將深入解析golang各種類(lèi)型是如何進(jìn)行比較的,需要的朋友可以參考下2024-01-01
Golang timer可能造成的內(nèi)存泄漏問(wèn)題分析
本文探討了Golang中timer可能造成的內(nèi)存泄漏問(wèn)題,通過(guò)分析一段代碼,解釋了為什么協(xié)程在調(diào)用timer.Stop()后無(wú)法正常退出,文章指出,timer.Stop()并不關(guān)閉Channel,導(dǎo)致協(xié)程無(wú)法繼續(xù)執(zhí)行,最后,提出了一種修復(fù)方法,并呼吁大家關(guān)注和分享2024-12-12
使用Go語(yǔ)言編寫(xiě)一個(gè)毫秒級(jí)生成組件庫(kù)文檔工具
在開(kāi)發(fā)組件庫(kù)的過(guò)程中,文檔無(wú)疑是不可或缺的一環(huán),在本文中將嘗試將Go語(yǔ)言與前端技術(shù)巧妙融合,以創(chuàng)建一款能在毫秒級(jí)別完成文檔生成的工具,需要的可以參考下2024-03-03

