
在go语言的并发模型中,goroutine是轻量级的执行单元。go运行时调度器负责管理这些goroutine的执行。runtime.gosched()是一个关键函数,它的作用是显式地通知go调度器,当前goroutine愿意让出cpu,以便其他goroutine有机会运行。
考虑以下Go程序示例:
package main
import (
"fmt"
"runtime"
)
func say(s string) {
for i := 0; i < 5; i++ {
runtime.Gosched() // 显式让出CPU
fmt.Println(s)
}
}
func main() {
go say("world") // 启动一个Goroutine
say("hello") // 在主Goroutine中执行
}当上述代码执行时,其输出通常是“hello”和“world”交替出现:
hello world hello world hello world hello world hello
这表明两个Goroutine(一个打印“hello”,一个打印“world”)轮流获得了执行机会。然而,如果我们将runtime.Gosched()这一行代码移除:
package main
import (
"fmt"
// "runtime" // runtime包不再被显式使用,可省略
)
func say(s string) {
for i := 0; i < 5; i++ {
// runtime.Gosched() // 移除让出调用
fmt.Println(s)
}
}
func main() {
go say("world")
say("hello")
}此时,程序的输出将变为:
hello hello hello hello hello
“world”从未被打印出来。这是因为在Go 1.5版本之前,当GOMAXPROCS环境变量未设置或设置为1时,Go运行时默认只使用一个操作系统线程来调度所有Goroutine。在这种单线程模式下,如果一个Goroutine不主动让出CPU,它就会一直占用执行权,导致其他Goroutine无法运行。runtime.Gosched()正是提供了这种显式让出机制,使得调度器能够切换上下文,让另一个Goroutine得以执行。
Go的Goroutine实现了一种“绿色线程”(green threads)或“协程”(coroutines)的概念,它们不直接映射到操作系统线程,而是由Go运行时管理和调度。理解Go调度器的工作方式需要区分两种多任务处理模型:
协作式多任务(Cooperative Multitasking):在这种模型下,任务(或Goroutine)必须主动让出CPU控制权,调度器才能切换到其他任务。如果一个任务未能及时让出,它可能会独占CPU,导致其他任务“饥饿”。在Go早期版本(特别是在GOMAXPROCS=1的默认设置下),Goroutine的调度很大程度上依赖于这种协作机制,例如通过使用并发原语(如channel操作)或显式调用runtime.Gosched()来让出。
抢占式多任务(Preemptive Multitasking):这是大多数现代操作系统线程所采用的模型。调度器可以在任何时候中断一个正在运行的任务,并切换到另一个任务,而无需任务主动配合。这使得任务之间的执行更加公平,也避免了单个任务长时间占用CPU的问题。
Go调度器从设计之初就致力于提供高效的并发能力,并随着版本迭代不断向更智能、更接近抢占式的方向发展。
GOMAXPROCS是一个重要的环境变量或运行时函数参数,它控制着Go运行时可以使用的最大操作系统线程数。
GOMAXPROCS = 1:当GOMAXPROCS设置为1时,Go运行时将所有Goroutine调度到一个单独的操作系统线程上。在这种情况下,如前所述,Goroutine的调度行为更倾向于协作式。如果一个Goroutine不显式让出,或者不执行会触发调度的操作(如channel通信、系统调用),它就可能独占CPU。
GOMAXPROCS > 1:当GOMAXPROCS设置为大于1的数值N时,Go运行时可以创建并使用最多N个操作系统线程来执行Goroutine。这意味着不同的Goroutine可能在不同的操作系统线程上并行运行(如果底层硬件支持多核)。在这种情况下,操作系统的抢占式调度机制会介入,管理这些操作系统线程的执行,从而间接为Goroutine提供了更强的抢占性。
你可以在程序中通过runtime.GOMAXPROCS()函数来设置这个值,例如:
package main
import (
"fmt"
"runtime"
)
func say(s string) {
for i := 0; i < 5; i++ {
// runtime.Gosched() // 在GOMAXPROCS > 1 时,Gosched的作用会减弱
fmt.Println(s)
}
}
func main() {
runtime.GOMAXPROCS(2) // 设置Go运行时使用2个OS线程
go say("world")
say("hello")
}当GOMAXPROCS设置为大于1时,即使移除了runtime.Gosched(),你也可能会观察到“hello”和“world”交错打印的现象,但其交错的模式可能是随机和不均匀的:
hello hello world hello world world hello ...
这种不确定性是多线程并行执行的典型特征。由于操作系统调度器在多个CPU核心上并行调度这些Go创建的OS线程,各个Goroutine的执行顺序变得不可预测。在这种情况下,runtime.Gosched()的作用会显著减弱,因为它不再是调度器切换上下文的唯一或主要方式。
Go语言的调度器在1.5版本之后经历了重要的改进,使其行为更加智能和高效:
因此,在现代Go版本中,runtime.Gosched()的必要性大大降低。调度器能够更自主地管理Goroutine的执行,使得并发程序的行为更加健壮和可预测(在宏观层面,但在微观执行顺序上仍具有不确定性)。
总之,runtime.Gosched()是Go语言并发模型中一个基础而重要的函数,尤其在Go早期版本和特定GOMAXPROCS设置下,它对于实现Goroutine之间的公平调度至关重要。随着Go语言的不断发展,调度器变得越来越智能,GOMAXPROCS的默认行为也得到了优化,使得现代Go程序在并发执行时通常不再需要显式调用runtime.Gosched(),但理解其背后的机制对于深入掌握Go并发编程仍然是不可或缺的。
以上就是Go并发调度:深入理解runtime.Gosched与GOMAXPROCS的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号