有时候可能会出现这种情况,一个无耻的goroutine阻止其他goroutine运行。当你有一个不让调度器运行的 for循环时,这就会发生。
package main
import "fmt"
func main() {
done := false
go func(){
done = true
}()
for !done {
}
fmt.Println("done!")
}
for循环并不需要是空的。只要它包含了不会触发调度执行的代码,就会发生这种问题。
具体的实现都在src/runtime/proc.c里,而要完成主动抢占,Go是采用在stack上做标记(g->stackguard0=StackPreempt),每次函数调用的时候会检查是否需要抢占,于是要想真的抢占goroutine CPU,只能等它调用任意一个非内联的函数。
package main
import "fmt"
func main() {
done := false
go func(){
done = true
}()
for !done {
fmt.Println("not done!") //not inlined
}
fmt.Println("done!")
}
要想知道你在 for循环中调用的函数是否是内联的,你可以在“go build”或“go run”时传入“-m” gc标志(如, go build -gcflags -m)。
package main
import (
"fmt"
"runtime"
)
func main() {
done := false
go func(){
done = true
}()
for !done {
runtime.Gosched()
}
fmt.Println("done!")
}