新闻中心

Go Goroutine 调度深度解析:理解并发与并行执行的差异

2025-11-29
浏览次数:
返回列表

Go Goroutine 调度深度解析:理解并发与并行执行的差异

本文深入探讨 go 语言中 goroutine 的调度机制,解释为何在多 goroutine 场景下,实际的并行执行行为可能与预期不符。我们将阐明 go 运行时调度器的工作原理,特别是 gomaxprocs 参数对并发与并行执行的影响,并提供实践建议以实现更有效的并行处理。

1. Goroutine 与 Go 并发模型概述

Go 语言通过 Goroutine 提供了一种轻量级的并发机制。Goroutine 是由 Go 运行时管理的协程,相比传统的操作系统线程,它具有更小的内存开销和更快的创建、切换速度。当使用 go 关键字启动一个函数时,它就会在一个新的 Goroutine 中异步执行。

理解 Go 的并发模型,首先要区分“并发”(Concurrency)和“并行”(Parallelism):

  • 并发:指能够同时处理多个任务的能力,即使这些任务并非在同一时刻执行。它们可能在单个处理器核心上通过时间片轮转的方式交替执行。
  • 并行:指多个任务在多个处理器核心上真正地同时、同步执行。

Go 语言天生支持并发,但要实现真正的并行则需要满足一定的条件。

2. Go 运行时调度器的工作原理

Go 运行时包含一个复杂的调度器,负责将 Goroutine 映射到操作系统线程上执行。这个调度器采用了 M:N 调度模型,即 N 个 Goroutine 可以在 M 个操作系统线程上运行。其核心组件包括:

  • G (Goroutine):Go 应用程序中的并发执行单元。
  • M (Machine/OS Thread):操作系统线程,是实际执行代码的载体。
  • P (Processor):逻辑处理器,代表一个 Go 运行时上下文,它将 G 绑定到 M 上。每个 P 维护一个本地的 Goroutine 队列,当 M 完成当前 G 的执行后,会从 P 的队列中取出下一个 G 执行。当 P 的本地队列为空时,它会尝试从其他 P 的队列中“偷取” Goroutine。

2.1 GOMAXPROCS 的作用

GOMAXPROCS 是一个环境变量,也可以通过 runtime.GOMAXPROCS 函数在程序中设置。它控制了 Go 运行时可以同时使用的逻辑处理器 P 的数量。

GoEnhance GoEnhance

全能AI视频制作平台:通过GoEnhance AI让视频创作变得比以往任何时候都更简单。

GoEnhance 347 查看详情 GoEnhance
  • 默认值:在 Go 1.5 版本及之后,GOMAXPROCS 的默认值通常设置为机器的 CPU 核心数 (runtime.NumCPU())。但在更早的版本中或某些特定环境下,其默认值可能为 1。
  • GOMAXPROCS = 1 的影响:当 GOMAXPROCS 设置为 1 时,即使系统拥有多个 CPU 核心,Go 运行时也只会创建一个逻辑处理器 P。这意味着所有 Goroutine 都必须在一个操作系统线程(M)上轮流执行。在这种情况下,Goroutine 之间只能实现并发,而无法实现真正的并行执行。它们会分时共享 CPU 资源,表现为交替执行。
  • GOMAXPROCS > 1 的影响:当 GOMAXPROCS 设置为大于 1 的值时,Go 运行时会创建相应数量的逻辑处理器 P。这些 P 可以被多个操作系统线程 M 使用,从而允许 Goroutine 在多个 CPU 核心上同时执行,实现并行。

3. 实验现象分析:为何并行执行不如预期?

在提供的实验代码中,用户启动了两个计算密集型(CPU 密集型)的 Goroutine,每个 Goroutine 都重复调用 fib 函数。观察到的现象是,在大部分时间内,两个 Goroutine 都是以大块的形式交替执行,而不是预期的“并排”或细粒度交错执行,直到任务后期才出现更频繁的切换。

这种现象通常是 GOMAXPROCS 默认值为 1 时的一个典型表现。当 GOMAXPROCS=1 时:

  1. 单线程执行:Go 运行时只有一个逻辑处理器 P,所有 Goroutine 都在这一个 P 绑定的 M(操作系统线程)上执行。
  2. 长时间占用 CPU:fib 函数是一个纯粹的 CPU 密集型计算,它不会主动进行 I/O 操作,也不会调用 runtime.Gosched() 等函数来主动让出 CPU。这意味着一个 Goroutine 一旦开始执行 fib 计算,它会尽可能长时间地占用 CPU,直到被 Go 调度器强制抢占(preemption)。
  3. 抢占式调度:Go 调度器在 Goroutine 运行时会定期进行抢占检查。在 Go 1.14 之前,抢占主要发生在函数调用或循环的后向跳转点。从 Go 1.14 开始,Go 引入了非协作式抢占(asynchronous preemption),使得调度器可以在任意安全点抢占 Goroutine。然而,即使有了抢占,对于长时间运行的 CPU 密集型任务,调度器也可能允许一个 Goroutine 运行相当长一段时间,直到达到抢占点或时间片耗尽。
  4. “后期交错”现象:至于为何在任务后期出现更频繁的交错,这可能与 fib 函数在计算较大数字时所需的执行时间有关,或者仅仅是调度器在特定时刻的决策。调度器的行为是非确定性的,它会根据内部启发式算法和系统负载来决定何时切换 Goroutine。在 CPU 密集型任务中,如果一个 Goroutine 不主动让出 CPU,调度器倾向于让它运行一段时间,以减少上下文切换的开销。

4. 实现真正的并行执行

要让 Goroutine 在多核 CPU 上实现真正的并行执行,关键在于确保 Go 运行时能够利用多个逻辑处理器 P。最直接的方法是设置 GOMAXPROCS。

4.1 设置 GOMAXPROCS

你可以通过以下两种方式设置 GOMAXPROCS:

  • 通过代码设置 (推荐): 在 main 函数的开头调用 runtime.GOMAXPROCS 函数。通常,我们会将其设置为机器的 CPU 核心数,以充分利用硬件资源。

    package main
    
    import (
        "fmt"
        "runtime"
        "sync" // 用于等待 Goroutine 完成
    )
    
    // fib 计算第 n 个斐波那

以上就是Go Goroutine 调度深度解析:理解并发与并行执行的差异的详细内容,更多请关注其它相关文章!


# 后期  # 营销推广实践报告  # 山西SEO优化电池  # 陕西网站建设的方案模板  # 寻亲网站建设美丽图片  # 蕲春seo优化案例  # 酒店网站建设全包  # 射阳seo优化创新  # 网站推广营销主题名称  # 汝南企业推广营销招聘  # 江苏优势网站建设  # 能与  # 多核  # go  # 默认值  # 它会  # 是一个  # 长时间  # 设置为  # 多个  # 环境变量  # ai  # mac  # 处理器  # 操作系统 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: Win11怎么安装Linux子系统 Win11 WSL2安装Ubuntu及环境配置指南  Yandex官网免登录入口_俄罗斯Yandex搜索引擎一键访问  在J*a中如何开发在线活动报名与管理系统_活动报名管理项目实战解析  蛙漫官方正版入口 蛙漫网页在线全集免费观看  Golang如何使用new_Go new分配内存机制讲解  支付宝如何设置安全保护_支付宝安全设置的全面教程  sublime怎么进行远程开发编辑_配置rsub/rmate实现sublime编辑服务器文件  win11怎么查看应用耗电情况 Win11电池设置查看应用能耗排行榜【优化】  怎样在Excel中做仪表盘_Excel仪表盘设计与关键指标展示方法  mcjs网页版流畅运行 mcjs低配电脑畅玩入口  QQ邮箱正确登录入口_QQ邮箱官方网站使用地址  Golang如何优化内存分配与垃圾回收_Golang内存管理与GC优化实践  特斯拉自动驾驶房车计划曝光 原型车将于2027年亮相  Win11 USB传输速度慢怎么解决 Win11 USB驱动更新与设置  双系统安装时,如何设置默认启动系统? msconfig命令了解一下!  解决Rails应用中内容错位与Turbo警告:meta标签误用导致富文本渲染异常  Win11输入法不见了怎么办_Windows11恢复语言栏显示方法  中兴Axon42Ultra怎样在文件App筛图_iPhone中兴Axon42Ultra文件App筛图【图片筛选】  html5 app怎么运行环境_配html5 app运行环境【教程】  期待已久:小米17 Ultra、小米首款NAS本月登场  Win11 BitLocker密码忘了怎么办 Win11找回BitLocker恢复密钥方法【解决】  mcjs网页版在线存档 mcjs云存档登录入口  12306选座如何查看座位示意图_12306座位示意图解读与使用  《明末:渊虚之羽》设计师谈设计角色:那会刚毕业 充满激情  c++如何使用Catch2编写单元测试_c++简洁易用的BDD风格测试框架  提升屏幕阅读器对“m”时间单位的播报准确性:HTML与CSS组合解决方案  如何将HTML表格多行数据保存到Google Sheets  Django模型中自动计算可用余额的实现方法  J*aScript中管理异步API调用:确保操作顺序与数据一致性  ArrayList与LinkedList操作复杂度详解:遍历与修改  精准捕获:如何在页面中监听除特定元素外的所有点击事件  妖精动漫免费平台 妖精动漫官网资源观看网址  iCloud登录入口网页版 苹果iCloud官网登录  品牌机怎么重装系统 联想/戴尔/惠普笔记本恢复出厂系统教程  b站如何看历史记录_b站观看历史找回方法  谷歌浏览器最新官方入口链接 谷歌浏览器网页版官网导航  Python自定义类排序:解决lambda键值访问TypeError的实践指南  漫画星球免费下拉式入口 漫画星球免费漫画在线阅读网站  word中如何让数字纵向排列_Word数字纵向排列方法  怎么去除衣服上的口红印_生活小妙招教你用酒精轻松擦除  CSS如何设置hover状态颜色_hover伪类调整背景或文字颜色  Golang如何使用net/url解析URL_Golang URL解析与处理方法  虫虫漫画精品漫画官网_虫虫漫画精品漫画官网进入精品漫画  谷歌邮箱网页版官方页面入口 谷歌邮箱网页端快速访问  win11如何卸载Windows更新补丁 Win11解决更新导致系统不稳定的问题【修复】  优化 Jest 模拟:强制未实现函数抛出错误以提升测试效率  Kafka Streams中基于消息头条件过滤消息的实现指南  蓝湖怎样用切图标注提对接效率_蓝湖用切图标注提对接效率【设计对接】  如何在Promise链中有效终止错误处理后的执行  微博网页版怎么开启两步验证_微博网页版账号安全两步验证设置方法 

搜索