转载地址:https://mp.weixin.qq.com/s/_By9rjPgb8zqIpVrsJrToQ

Illustration created for “A Journey With Go”, made from the original Go Gopher, created by Renee French.

ℹ️ 这篇文章基于 Go 1.13 版本。

在 Go 中创建 gorotine 既方便又快捷,然而 Go 在同一时间内最多在一个核上运行一个 gorotine,因此需要一种方法来存放其他的 gorotine,从而确保处理器(processor)负载均衡。

Goroutine 队列

Go 使用两级队列来管理等待中的 goroutine,分别为本地队列和全局队列。每一个处理器都拥有本地队列,而全局队列是唯一的,且能被所有的处理器访问到:

Global and local queues

每个本地队列都有最大容量,为 256。在容量满了之后,任意新到来的 Goroutine 都会被放置到全局队列。下面的例子是,生产了上千个 Goroutine 的程序:

func main() {var wg sync.WaitGroupfor i := 0;i < 2000 ;i++ {wg.Add(1)Go func() {a := 0for i := 0; i < 1e6; i++ {a += 1}wg.Done()}()}wg.Wait()
}

下面是拥有两个处理器的调度器追踪数据(traces):

Details of the local and global queues

追踪数据通过 runqueue 展示了全局队列中 Goroutine 的数量,以及方括号中 [3 256] 的本地队列 goroutine 数量(分别为 P0 和 P1)。当本地队列满了,积压了 256 个等待中的 goroutine 后,下一个 Goroutine 会被压栈到全局队列中,正如我们从 runqueue 看到的数量增长一样。

Goroutine 仅在本地队列满载之后才会加入到全局队列;它也会在 Go 往调度器中批量注入时被加到全局队列,例如,网络轮询器(network poller) 或者在垃圾回收期间等待的 goroutine。

下面是上一个例子的图示:

Local queues have up to 256 goroutines

不过,我们还想知道,为什么本地队列 P0 在上一个列子中不为空。因为 Go 使用了其他策略确保每个处理器都有任务处理。

任务窃取

如果处理器没有任务可处理,它会按以下规则来执行,直到满足某一条规则:

  • 从本地队列获取任务

  • 从全局队列获取任务

  • 从网络轮询器获取任务

  • 从其它的处理器的本地队列窃取任务

在我们前面的例子中,主函数在 P1 上运行并创建 goroutine。当第一批 gourinte 已经进入了 P1 的本地队列时,P0 正在寻找任务。然而,它的本地队列,全局队列,以及网络轮询器都是空的。最后的解决方法是从 P1 中窃取任务。

Work-stealing by P0

下面是调度器在发生任务窃取前后的追踪数据:

Work-stealing by P0

追踪数据展示了,处理器是如何从其它处理器中窃取任务的。它从(其他处理器的)本地队列中取走一半的 goroutine;在七个 Goroutine 中,偷走了四个 —— 其中一个立马在 P0 执行,剩下的放到本地队列。现在处理器间工作处于负载良好的状态。这能通过执行 tracing 来确认:

goroutine 被合理地分发,然后因为没有 I/O,goroutine 被链式执行而不需要切换。我们现在看一下,当出现例如涉及到文件操作等 I/O 时,会发生什么。

I/O 与全局队列

一起看下涉及到文件操作的例子:

func main() {var wg sync.WaitGroupfor i := 0;i < 20 ;i++ {wg.Add(1)Go func() {a := 0for i := 0; i < 1e6; i++ {a += 1if i == 1e6/2 {bytes, _ := ioutil.ReadFile(`add.txt`)inc, _ := strconv.Atoi(string(bytes))a += inc}}wg.Done()}()}wg.Wait()
}

变量 a 随着时间以文件的字节数增加,下面是新的追踪数据:

在这个例子中,我们能看到每一个 Goroutine 不只被一个处理器处理。在系统调用的情况下,当调用完成后,Go 使用网络轮询器从全局队列中把 gouroutine 取回来。这里是 Goroutine #35 的一个示意图:

I/O operations put the work back to the global queue

当一个处理器能从全局队列中获取任务,第一个可用的处理器( P) 会执行这个 goroutine。这个行为解释了,为什么一个 Goroutine 能在不同的处理器中运行,也展示了 Go 是如何让空闲的处理器资源运行 goroutine,从而进行系统调用的优化。


via: https://medium.com/a-journey-with-go/go-work-stealing-in-go-scheduler-d439231be64d

作者:Vincent Blanchon[1]译者:LSivan[2]校对:polaris1119[3]

本文由 GCTT[4] 原创编译,Go 中文网[5] 荣誉推出

参考资料

[1]

Vincent Blanchon: https://medium.com/@blanchon.vincent

[2]

LSivan: https://github.com/LSivan

[3]

polaris1119: https://github.com/polaris1119

[4]

GCTT: https://github.com/studygolang/GCTT

[5]

Go 中文网: https://studygolang.com/


推荐阅读

  • go trace 剖析 go1.14 异步抢占式调度

都说 Go 可以开启成千上万的 Goroutine,那调度器是怎么处理核上任务分配的?相关推荐

  1. [转]Golang中goroutine的调度器详解

    Go调度器原理浅析 来源:https://www.douban.com/note/300631999/ goroutine是golang的一大特色,或者可以说是最大的特色吧(据我了解),这篇文章主要翻 ...

  2. Goroutine 并发调度模型深度解析之手撸一个高性能 goroutine 池

    文章目录 1 前言 2 Goroutine & Scheduler 2.1 线程那些事儿 2.1.1 用户级线程模型 2.1.2 内核级线程模型 2.1.3 两级线程模型 2.2 G-P-M ...

  3. Goroutine并发调度模型深度解析之手撸一个协程池

    Goroutine & Scheduler Goroutine,Go语言基于并发(并行)编程给出的自家的解决方案.goroutine是什么?通常goroutine会被当做coroutine(协 ...

  4. c++ 初始化 代码 应放在那里_Go语言goroutine调度器初始化 (12)

    先吐槽一下,知乎编辑器居然不支持汇编语言,代码的空格也给我弄没了,你说你把运算符两边的空格搞掉就搞掉吧,还能看,你为啥要把if, for后面的空格也搞掉啊... 本文是<Go语言调度器源代码情景 ...

  5. 【STM32】FreeRTOS 调度器开启和任务相关函数详解

    文章目录 调度器开启过程分析 调度器开启过程分析 前面的所有例程中我们都是在 main()函数中先创建一个开始任务 start_task,后面紧接着调 用函数 vTaskStartScheduler( ...

  6. Goroutine调度器及面试精选

    Goroutine调度器 Go语言在并发编程有着非常强大的能力,讲到调度器,我们的话题离不开操作系统.进程与线程这些概念,在学习操作系统时,线程是操作系统调度的最基本单元. 在没有学习Go语言之前,线 ...

  7. go gmp --- goroutine抢占调度源码分析

    为什么需要goroutine抢占调度? 通过本文我们已经知道,go会在每次调度goroutine的时候检查定时器是否就绪了,所以如果仅靠goroutine主动让出(系统调用阻塞.读写chan阻塞-)而 ...

  8. 计算机桌面的图片要点开才可以看,我的电脑直接双击图片打不开,需要右键开启方式里选择图片检视器才可以,请问有什么方法修改一下?...

    我的电脑直接双击图片打不开,需要右键开启方式里选择图片检视器才可以,请问有什么方法修改一下?以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容,让我们赶 ...

  9. Go goroutine 的调度模型及设置运行CPU数目

    goroutine 的调度模型 MPG 模式基本介绍 M:操作系统的主线程(是物理线程) P:协程执行需要的上下文环境 G:协程 MPG 模式运行的状态 1 当前程序有三个M,如果三个M都在一个cpu ...

最新文章

  1. hadoop系统 hdfs 命令行操作
  2. 如何查看SharePoint未知错误
  3. 我的编程生涯的入门语言 - C语言之学员成绩管理
  4. Hibernate上传数据到数据库,从数据库读取数据到本地模板代码
  5. 我用过的一些web.xml配置
  6. eclipse/myeclipse高亮显示相同变量名 .
  7. html font-family设置无效,css设置中文字体(font-family:黑体)后样式失效问题
  8. 光模块兼容性介绍,如何测试兼容光模块的兼容性?
  9. 95-280-046-源码-资源管理-磁盘
  10. python 员工考勤_用python写的考勤自动打卡程序
  11. Julia: Join与字符串Array
  12. linux下keytool生成证书_keytool命令 – 密钥和证书管理工具
  13. css制作三角形、带三角文本框、价格三角框
  14. python语言程序设计基础课后习题答案
  15. JUnit测试提示java.lang.Exception: No runnable methods
  16. CTGNet GIA和CN2 GIA的区别
  17. 企业在做搜索引擎优化时应该注意什么?
  18. 一分钟了解“#include命令是干啥的”
  19. 大数据-Hadoop-云服务器的搭建
  20. 线程条件队列ConditionObject源码解读

热门文章

  1. Nginx+Tpmcat 负载均衡
  2. java中对date的一些处理以及获取date
  3. 金蝶k3rpc服务器不可用_金蝶KIS商贸版常见问题这样解决
  4. 手机必备OCR文字识别软件:福昕扫描王使用攻略
  5. 一线互联网企业面试题总结(帮你成功拿到offer)
  6. 【WEB2.0】 网页调用QQ聊天(PC+M站)
  7. phalcon: Profiling分析 profilter / Plugin结合,dispatcher调度控制器 监听sql执行日志
  8. 20150318知识小结
  9. 炼数成金hadoop视频干货03
  10. Microsoft Visual Studio 2008从试用版转为正式版