新闻中心
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
全能AI视频制作平台:通过GoEnhance AI让视频创作变得比以往任何时候都更简单。
347
查看详情
- 默认值:在 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 时:
- 单线程执行:Go 运行时只有一个逻辑处理器 P,所有 Goroutine 都在这一个 P 绑定的 M(操作系统线程)上执行。
- 长时间占用 CPU:fib 函数是一个纯粹的 CPU 密集型计算,它不会主动进行 I/O 操作,也不会调用 runtime.Gosched() 等函数来主动让出 CPU。这意味着一个 Goroutine 一旦开始执行 fib 计算,它会尽可能长时间地占用 CPU,直到被 Go 调度器强制抢占(preemption)。
- 抢占式调度:Go 调度器在 Goroutine 运行时会定期进行抢占检查。在 Go 1.14 之前,抢占主要发生在函数调用或循环的后向跳转点。从 Go 1.14 开始,Go 引入了非协作式抢占(asynchronous preemption),使得调度器可以在任意安全点抢占 Goroutine。然而,即使有了抢占,对于长时间运行的 CPU 密集型任务,调度器也可能允许一个 Goroutine 运行相当长一段时间,直到达到抢占点或时间片耗尽。
- “后期交错”现象:至于为何在任务后期出现更频繁的交错,这可能与 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链中有效终止错误处理后的执行
微博网页版怎么开启两步验证_微博网页版账号安全两步验证设置方法


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