首頁 >後端開發 >Golang >當並發 goroutine 遞增共享變數時,為什麼我的 Go 程式的輸出始終為 1?

當並發 goroutine 遞增共享變數時,為什麼我的 Go 程式的輸出始終為 1?

Susan Sarandon
Susan Sarandon原創
2024-10-31 01:36:29664瀏覽

Why is the output of my Go program always 1 when a concurrent goroutine is incrementing a shared variable?

這是因為 go 編譯器最佳化了程式碼嗎?

在提供的 Go 程式碼中,變數 i 在 main goroutine 和 goroutine 之間共用使用 go func() { ... }() 建立的並發 goroutine。並發 Goroutine 的目的是無限增加 i。然而,在運行程式時,您注意到輸出始終為 1,儘管預期的數字要大得多,因為並發 goroutine 有足夠的時間將 i 遞增多次。

要理解這種行為,我們需要考慮 Go 記憶體模型和編譯器的最佳化。

Go 記憶體模型

Go 記憶體模型指定了可以保證在一個 goroutine 中讀取變數的條件觀察不同 Goroutine 中寫入同一變數所產生的值。

根據 Go 記憶體模型,為了使共享變數的變更在 Goroutine 之間可見,它們後面必須跟有同步事件,例如通道操作或鎖定互斥鎖。

編譯器最佳化

在您的範例中,並發 goroutine 中對 i 的分配後面沒有任何同步事件。因此,不能保證主 Goroutine 會觀察到賦值,並且編譯器可以根據需要自由地最佳化程式碼。

激進的編譯器可能會透過完全消除增量操作來優化程式碼,有效地將並發 goroutine 減少為不執行任何操作的無限循環。這種最佳化可以解釋為什麼輸出總是 1,因為主 Goroutine 從不觀察並發 Goroutine 執行的增量。

為了確保共享變數的更新能夠在 Goroutine 之間正確傳播,同步存取非常重要使用 Go 的同步和同步/原子包提供的通道、互斥體或其他同步原語來存取這些變數。

以上是當並發 goroutine 遞增共享變數時,為什麼我的 Go 程式的輸出始終為 1?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn