Golang常見錯誤之值拷貝和for循環(huán)中的單一變量詳解
前言
golang(中文名:go語言)是谷歌2009發(fā)布的第二款開源編程語言。Go語言專門針對多處理器系統(tǒng)應用程序的編程進行了優(yōu)化,使用Go編譯的程序可以媲美C或C++代碼的速度,而且更加安全、支持并行進程。。如果你想知道得更多,請移步至官網(wǎng)golang官網(wǎng)
在 Go 中函數(shù)的調(diào)用是值拷貝 copy value,而且在 for 循環(huán)中 v 的變量始終是一個變量。如果 v 是 pointer,print 這個 method 接收的是指針的拷貝,for 循環(huán)體中每次迭代 v 的 pointer value 都是不同的,所以輸出不同。
在 Go 常見的錯誤一文中 http://devs.cloudimmunity.com/gotchas-and-common-mistakes-in-go-golang/ 有這么一段代碼:
package main import ( "fmt" "time" ) type field struct { name string } func (p *field) print() { fmt.Println(p.name) } func main() { data := []field{{"one"},{"two"},{"three"}} for _,v := range data { go v.print() } time.Sleep(3 * time.Second) //goroutines print: three, three, three }
把 field slice 的類型改為 pointer 結(jié)果又不同:
package main import ( "fmt" "time" ) type field struct { name string } func (p *field) print() { fmt.Println(p.name) } func main() { data := []*field{{"one"},{"two"},{"three"}} for _,v := range data { v := v go v.print() } time.Sleep(3 * time.Second) //goroutines print: one, two, three }
這兩段代碼的差異究竟是如何導致結(jié)果的不同?
我對上面的代碼 for 循環(huán)中的部分進行了一下改造,改造之后對應的代碼分別是:
slice 是非指針
data := []field{{"one"},{"two"},{"three"}} for _,v := range data { pp := (*field).print go pp(&v) //非 pointer }
slice 是指針
data := []*field{{"one"},{"two"},{"three"}} for _,v := range data { pp := (*field).print go pp(v) // pointer }
改造之后再去看原來的代碼就能看出最明顯的差異在 print 的這個 method 的 receiver 的傳遞上。
在 Go 中函數(shù)的調(diào)用是值拷貝 copy value,而且在 for 循環(huán)中 v 的變量始終是一個變量。
如果 v 是 pointer,print 這個 method 接收的是指針的拷貝,for 循環(huán)體中每次迭代 v 的 pointer value 都是不同的,所以輸出不同。
如果 v 是一個普通的 struct,for 循環(huán)體中每次迭代 &v 都是 v 這個變量本身的 pointer,也就是總是指向同一個 field,由于在很大程度上這段代碼中的 goroutine 都是在 for 結(jié)束之后才執(zhí)行,而此時 v 將會指向最后一個 field,也就是 {"three"},所以輸出相同。
有人說 one、two、three 的隨機輸出是因為 CPU 是多核的原因?qū)е碌?,如果改成單核就是順序輸出,這樣的說法并不是特別準確。理論上來講 goroutine 的調(diào)度是有一定的隨機性的,也就是即使是單核輸出也有可能是隨機的,只是在運行如此簡單的例子時一般機器環(huán)境都不會導致這 3 個簡單的 goroutine 出現(xiàn)交叉執(zhí)行。比如可以在 print 輸出之前模擬 io 繁忙的來達到即使是單核也可能是隨機輸出的目的。
if rand.Intn(100) > 20 { time.Sleep(1 * time.Second) }
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關文章
golang jsoniter extension 處理動態(tài)字段的實現(xiàn)方法
這篇文章主要介紹了golang jsoniter extension 處理動態(tài)字段的實現(xiàn)方法,我們使用實例級別的 extension, 而非全局,可以針對不同業(yè)務邏輯有所區(qū)分,jsoniter 包提供了比較完善的定制能力,通過例子可以感受一下擴展性,需要的朋友可以參考下2023-04-04Golang實現(xiàn)自己的Redis數(shù)據(jù)庫內(nèi)存實例探究
這篇文章主要為大家介紹了Golang實現(xiàn)自己的Redis數(shù)據(jù)庫內(nèi)存實例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2024-01-01golang中的string與其他格式數(shù)據(jù)的轉(zhuǎn)換方法詳解
這篇文章主要介紹了golang中的string與其他格式數(shù)據(jù)的轉(zhuǎn)換方法,文章通過代碼示例介紹的非常詳細,對大家的學習或工作有一定的幫助,需要的朋友可以參考下2023-10-10golang常用庫之字段參數(shù)驗證庫-validator使用詳解
這篇文章主要介紹了golang常用庫:字段參數(shù)驗證庫-validator使用,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借價值,需要的朋友可以參考下2020-10-10pytorch中的transforms.ToTensor和transforms.Normalize的實現(xiàn)
本文主要介紹了pytorch中的transforms.ToTensor和transforms.Normalize的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-04-04