当从下面的代码中删除fmt.Print()
行时,代码无限运行。为什么呢?
package main
import "fmt"
import "time"
import "sync/atomic"
func main() {
var ops uint64 = 0
for i := 0; i < 50; i++ {
go func() {
for {
atomic.AddUint64(&ops, 1)
fmt.Print()
}
}()
}
time.Sleep(time.Second)
opsFinal := atomic.LoadUint64(&ops)
fmt.Println("ops:", opsFinal)
}
答案 0 :(得分:8)
Go By Example article includes:
// Allow other goroutines to proceed.
runtime.Gosched()
fmt.Print()
扮演类似的角色,并允许main()
有机会继续。
export GOMAXPROCS=2
可能有助于程序在无限循环的情况下完成,如&#34; golang: goroute with select doesn't stop unless I added a fmt.Print()
&#34;中所述。
fmt.Print()
明确地将控制传递给某些系统调用
是的,go1.2+ has pre-emption in the scheduler
在以前的版本中,永远循环的goroutine可能会在同一个线程上饿死其他goroutine,这是
GOMAXPROCS
只提供一个用户线程时的一个严重问题。在Go 1.2中,部分解决了这个问题:在进入函数时偶尔会调用调度程序。这意味着任何包含 a(非内联)函数调用的循环都可以被抢占,允许其他goroutine在同一个线程上运行。
注意重点(我提出):在你的例子中,有可能是for循环atomic.AddUint64(&ops, 1)
。那里没有先发制人。
2017年更新:Go 1.10 will get rid of GOMAXPROCS
。