GMP调度模型到底是怎么回事?手绘图解最全版

发布时间 - 2026-01-23 00:00:00    点击率:
GMP调度模型是Go运行时的用户态并发调度机制,由G(goroutine)、M(OS线程)、P(逻辑处理器)协同实现高效复用与负载均衡。

GMP调度模型是Go语言运行时实现并发的核心机制,它用三个关键角色(G、M、P)协同工作,让成千上万个goroutine能在少量操作系统线程上高效、公平地运行。它不是操作系统级的调度,而是Go自己在用户态做的“二次调度”,目标是减少系统调用开销、提升CPU利用率、隐藏I/O等

待时间。

三个主角:G、M、P 分别是什么?

G(Goroutine):轻量级协程,Go里的并发执行单元。每个G只占2KB栈空间(可动态伸缩),创建销毁成本极低。它不绑定线程,只保存自己的执行上下文(如PC、栈指针、寄存器等)。

M(Machine):对应一个OS线程(比如Linux上的pthread)。M负责真正执行G的代码。M数量默认无硬上限,但受P数量和阻塞操作影响——当M因系统调用或锁等待而阻塞时,Go运行时会尝试复用其他M。

P(Processor):逻辑处理器,是G调度的关键枢纽。P的数量默认等于CPU核心数(可通过GOMAXPROCS设置),它持有:
• 本地G队列(最多256个,FIFO)
• 全局G队列(所有P共享,有锁)
• 可运行G的“所有权”——只有拥有P的M才能执行G
• 内存分配缓存(mcache)、垃圾回收状态等

调度是怎么流动起来的?

一个G从诞生到执行,经历典型的“入队→获取P→被M执行→可能让出/阻塞→再调度”过程:

  • 新G创建后,优先加入当前P的本地队列;若本地队列满,则入全局队列
  • M需绑定一个P才能运行G:启动时抢一个空闲P;若M阻塞(如syscall),会释放P供其他M使用
  • M从自己绑定的P中按顺序取G执行;若本地队列空,先尝试从全局队列偷一批(至少1/4),再尝试“工作窃取”(steal)——从其他P的本地队列尾部偷一半
  • G主动让出(如调用runtime.Gosched())、被抢占(如超过10ms的连续运行,由sysmon线程触发)、或进入阻塞(如channel收发、网络读写、time.Sleep),都会触发调度器介入,保存现场并寻找下一个可运行G

几个关键设计细节,决定它为什么快又稳

非抢占式 + 有限抢占:Go早期完全协作式,现在通过sysmon监控线程,对长时间运行的G强制插入抢占点(如函数入口、循环回边),避免某个G饿死其他G。

系统调用处理巧妙:M进入阻塞系统调用前,会将P解绑并转入自旋状态(尝试找新M接管);调用返回后,M优先尝试重新获取原P;失败则把G放回全局队列,自己休眠——避免P空转、也避免M堆积。

内存与GC友好:每个P自带mcache,小对象分配无需锁;GC的三色标记也以P为单位并发扫描,提升停顿控制能力。

一张图看懂核心流转(文字版示意)

(想象手绘草图结构)
顶部:多个P(P0, P1, P2…)并排,每个P下挂一个“本地G队列”(竖列小方块G1-G5)
中间:若干M(M0-M3)箭头指向各自绑定的P;M0正在执行P0的G1,M1空闲等待P,M2卡在syscall(画个云朵标“阻塞中”)
底部:一个“全局G队列”+“netpoller”(处理异步网络事件)+“sysmon”(小哨兵图标,监视M和G)
箭头穿插:G从全局队列→P本地队列;M从P偷G;阻塞M释放P→空闲M抢P;sysmon向G发送抢占信号

不复杂但容易忽略:GMP不是静态绑定,而是一套动态借还、平衡负载的协作协议。理解它,能帮你写出更少竞争、更低延迟、更易诊断的Go并发代码。


# linux  # go  # 操作系统  # 处理器  # go语言  # mac  # gmp  #   # 为什么  # 有锁  # 循环  # 指针  #   # 线程 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 湖南网站制作公司,湖南上善若水科技有限公司做什么的?  如何在阿里云高效完成企业建站全流程?  英语简历制作免费网站推荐,如何将简历翻译成英文?  Laravel路由怎么定义_Laravel核心路由系统完全入门指南  iOS UIView常见属性方法小结  微信小程序制作网站有哪些,微信小程序需要做网站吗?  Android利用动画实现背景逐渐变暗  如何用wdcp快速搭建高效网站?  Python结构化数据采集_字段抽取解析【教程】  使用C语言编写圣诞表白程序  晋江文学城电脑版官网 晋江文学城网页版直接进入  HTML透明颜色代码怎么让图片透明_给img元素加透明色的技巧【方法】  网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?  如何在 Python 中将列表项按字母顺序编号(a.、b.、c. …)  高性价比服务器租赁——企业级配置与24小时运维服务  javascript如何操作浏览器历史记录_怎样实现无刷新导航  Laravel怎么生成二维码图片_Laravel集成Simple-QrCode扩展包与参数设置【实战】  Laravel如何生成URL和重定向?(路由助手函数)  如何用5美元大硬盘VPS安全高效搭建个人网站?  米侠浏览器网页背景异常怎么办 米侠显示修复  Laravel如何为API编写文档_Laravel API文档生成与维护方法  香港服务器WordPress建站指南:SEO优化与高效部署策略  米侠浏览器网页图片不显示怎么办 米侠图片加载修复  如何快速上传自定义模板至建站之星?  如何获取免费开源的自助建站系统源码?  宙斯浏览器视频悬浮窗怎么开启 边看视频边操作其他应用教程  大型企业网站制作流程,做网站需要注册公司吗?  网站制作大概要多少钱一个,做一个平台网站大概多少钱?  JavaScript Ajax实现异步通信  北京网页设计制作网站有哪些,继续教育自动播放怎么设置?  如何在阿里云完成域名注册与建站?  奇安信“盘古石”团队突破 iOS 26.1 提权  JavaScript实现Fly Bird小游戏  Laravel如何实现全文搜索_Laravel Scout集成Algolia或Meilisearch教程  黑客如何通过漏洞一步步攻陷网站服务器?  如何快速搭建高效服务器建站系统?  Laravel如何处理JSON字段_Eloquent原生JSON字段类型操作教程  如何快速启动建站代理加盟业务?  Laravel怎么多语言本地化设置_Laravel语言包翻译与Locale动态切换【手册】  手机钓鱼网站怎么制作视频,怎样拦截钓鱼网站。怎么办?  智能起名网站制作软件有哪些,制作logo的软件?  Laravel Fortify是什么,和Jetstream有什么关系  php嵌入式断网后怎么恢复_php检测网络重连并恢复硬件控制【操作】  如何快速搭建安全的FTP站点?  Laravel如何创建和注册中间件_Laravel中间件编写与应用流程  东莞专业网站制作公司有哪些,东莞招聘网站哪个好?  HTML透明颜色代码在Angular里怎么设置_Angular透明颜色使用指南【详解】  Windows驱动无法加载错误解决方法_驱动签名验证失败处理步骤  青岛网站建设如何选择本地服务器?  今日头条AI怎样推荐抢票工具_今日头条AI抢票工具推荐算法与筛选【技巧】