Goroutine 協作調度和執行順序
Go 中的Goroutines 是協作調度的,這意味著它們明確地將執行交給其他Goroutine,而不是被搶佔由作業系統。這就提出了一個問題:如果一個goroutine沒有顯式yield,是否會阻止其他goroutine運行?
連續循環的影響
根據文檔,運行的goroutine連續而不讓步會使同一線程上的其他 goroutine「挨餓」。這是因為永久執行循環的 goroutine 會阻止調度程序將時間片分配給其他 goroutine。
函數呼叫作為屈服點
但是,需要注意的是Goroutine 中的函數呼叫可以充當隱式的屈服點。具體來說,呼叫涉及系統資源的函數(例如網路輸入、睡眠、通道操作或阻塞原語)會觸發調度程序。
範例場景
考慮一個簡單的Go 程序,啟動四個goroutine,每個goroutine 執行一個循環並列印一個總和:
func sum(x int) { sum := 0 for i := 0; i < x; i++ { sum += i } fmt.Println(sum) }
如果這些goroutine 與以下同時啟動:
go sum(100) go sum(200) go sum(300) go sum(400)
goroutine不一定會一一執行。在每個 goroutine 末尾進行的 fmt.Println() 呼叫將觸發調度程序,從而允許其他 goroutine 運行。
附加說明
Go 運行時的最新改進引入了讓 goroutine 產生執行的額外方法。例如,存在一項提案,將調度程序呼叫插入到沒有任何函數呼叫的循環中。
總之,雖然沒有屈服的連續循環可能會導致單線程環境中的其他goroutine 挨餓,但函數調用和最近的運行時增強功能提供了即使沒有顯式讓步的情況下,也能實現goroutine 調度機制。
以上是Go 中連續的 Goroutine 循環會導致其他 Goroutines 飢餓嗎?的詳細內容。更多資訊請關注PHP中文網其他相關文章!