
go语言以其轻量级并发原语goroutine而闻名。goroutine是go运行时管理的并发执行单元,比传统操作系统线程更轻量,启动开销更小。go运行时调度器负责在操作系统线程上高效地调度这些goroutine。然而,理解go调度器的工作方式对于编写正确的并发程序至关重要,尤其是在goroutine之间需要协作执行的场景。
Go的调度器并非严格意义上的抢占式调度器(preemptive scheduler),它更多地依赖于协作式调度(cooperative scheduling)。这意味着一个Goroutine在执行时,通常需要主动让出CPU的控制权,调度器才有机会将CPU分配给其他等待执行的Goroutine。常见的让出控制权的操作包括:
考虑以下Go语言教程中的经典示例,它展示了两个Goroutine的并发执行:
package main
import (
"fmt"
"time"
)
func say(s string) {
for i := 0; i < 5; i++ {
time.Sleep(100 * time.Millisecond) // 关键的让渡点
fmt.Println(s)
}
}
func main() {
go say("world") // 启动一个并发Goroutine
say("hello") // 主Goroutine执行
}当运行这段代码时,输出会是"world"和"hello"交替出现5次。这是因为go say("world")启动了一个新的Goroutine,而say("hello")则在main函数所在的主Goroutine中执行。在say函数内部,每次循环都会调用time.Sleep(100 * time.Millisecond)。这个time.Sleep调用正是Goroutine让出CPU控制权的关键点。当say("hello") Goroutine执行到time.Sleep时,它会暂停100毫秒并让出CPU。此时,Go调度器就有机会切换到say("world") Goroutine,让它执行一部分代码,直到它也遇到time.Sleep并让出CPU。如此往复,便实现了两个Goroutine的交替执行。
现在,如果我们将say函数中的time.Sleep行注释掉,代码将变为:
立即学习“go语言免费学习笔记(深入)”;
package main
import (
"fmt"
// "time" // time包也不再需要导入
)
func say(s string) {
for i := 0; i < 5; i++ {
// time.Sleep(100 * time.Millisecond) // 已移除
fmt.Println(s)
}
}
func main() {
go say("world")
say("hello")
}重新运行这段代码,我们会发现屏幕上只打印了五次"hello",而"world"从未出现。这正是因为移除了time.Sleep,say("hello")所在的主Goroutine在执行其for循环时,不再有任何主动让出CPU控制权的操作。它会以极快的速度连续执行五次fmt.Println("hello"),在没有任何阻塞或让渡点的情况下完成其所有工作。
一旦主Goroutine中的say("hello")函数执行完毕,main函数也将随之结束。Go程序会在main函数返回后立即终止,而不会等待其他非主Goroutine完成。由于say("world") Goroutine在主Goroutine快速执行期间没有获得任何执行机会,它甚至可能还未开始执行,程序就已经退出了。因此,time.Sleep在这里的作用并非“拯救”Goroutine免于死亡,而是提供了必要的让渡点,确保了并发Goroutine有机会被调度器选中并执行。
虽然time.Sleep在教学示例中能清晰地展示Goroutine的协作调度,但在实际生产环境中,它通常不是控制Goroutine生命周期或同步执行的最佳实践。过度或不当使用time.Sleep可能导致性能问题或竞态条件。
更专业的Goroutine同步和等待机制包括:
注意事项:
time.Sleep在Go语言的Goroutine调度中扮演着一个重要的角色,尤其是在展示协作式调度机制时。它迫使当前Goroutine让出CPU控制权,从而允许Go调度器有机会切换到其他Goroutine执行。当缺少这种让渡机制时,如果主Goroutine迅速完成,那么其他并发Goroutine可能因为没有获得执行机会而“夭折”,导致程序未能按预期运行。因此,理解Goroutine的协作调度原理以及如何通过适当的机制(包括但不限于time.Sleep)来确保Goroutine的公平执行,是编写高效、正确Go并发程序的关键。
以上就是深入理解Go语言Goroutine调度:time.Sleep为何至关重要的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号