一文详解golang中的gmp模型

 更新时间:2023年11月15日 11:07:44   作者:wric  
这篇文章主要介绍了golang中的gmp模型的诞生、概念及调度讲解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

诞生

(我们为什么需要gmp模型)

我们知道,早期的操作系统操作系统是单进程的。后来引入了并发的思想,发展为多进程/多线程操作系统,操作系统可以通过时间片在多个进程之间切换。然而,在多个进程之间进行切换,是需要切换成本的。

在引入协程之前,我们需要回顾一下进程和线程的概念。
进程,是软件的执行副本,是分配资源的基础单位。每个进程有自己的进程控制块,代码,数据。
线程,是轻量级的线程,是调度和执行的基本单位。(这里单指内核态线程),每个线程有自己的寄存器和栈。

时间与空间

在传统的多线程并发模型中,轻量级的线程只需进行寄存器和栈等上下文的切换就可以完成多个任务的并发执行。然而,随着并发量越来越高,有些局限性开始展现出来。

随着并发量的上涨,切换的频率越来越高,线程的切换开销开始显露不足
随着并发量的上涨,需要越来越多的线程,内存占用也越来越大

因此,在高并发的需求下,协程因此诞生了,也就是golang中的groutine。

在时间上,平均每次协程切换的开销是 120ns 左右,相对于进程切换的开销大约 3.5us,大约是其的三十分之一
在空间上,协程初始化创建的时候为其分配的栈有 2KB,而线程栈一般在4-8M左右。

那么,为什么协程可以这么轻量呢。

在传统的多线程中,线程与进程的关系是固定的,每个线程必定从属于一个进程。而到了协程,为了去掉更多的枷锁,协程是不依赖于某个固定的线程的,协程可以自由切换而不比。换而言之,golang在用户空间与内核空间之间创建了一层新的映射,而这,就是gmp模型。

概念

(gmp是什么)

gmp=groutine+machine+processor

G

  • (1) g 即goroutine,是golang中对协程的抽象;
  • (2) g有自己的运行栈、状态、以及执行的任务函数(用户通过go func指定);
  • (3) g需要绑定到p才能执行,在g的视角中,p就是它的cpu.

M

  • (1) m 即machine,是golang 中对线程的抽象;
  • (2) m的数量是动态的,golang语言中默认最大值为10000,也可以用runtime/debug包中的SetMaxThreads函数来设置
  • (3) ⼀个M阻塞,会创建⼀个新的M。如果有M空闲,那么就会回收或者睡眠。

P

  • (1) p即 processor,是golang 中的调度器;
  • (2) p是 gmp的中枢,借由p承上启下,实现g 和m之间的动态有机结合;
  • (3)对g而言,p是其cpu,g只有被p调度,才得以执行;
  • (4)对m而言,p是其执行代理,为其提供必要信息的同时(可执行的g、内存分配情况等),并隐藏了繁杂的调度细节;
  • (5) p 的数量决定了g最大并行数量,可由用户通过GOMAXPROCS进行设定(超过CPu核数时无意义).也可以在程序中通过runtime.GOMAXPROCS() 来设置

gmp模型图片

调度

调度策略1 work steal
自己的本地队列为空时,会去别的本地队列偷取一半,保证有任务可以执行。

调度策略2 hand off
当一个绑定了M的G发生阻塞,会创建/唤醒

调度策略3 并行
最多有GOMAXPROCS个线程分布在多个CPU上同时运⾏

调度策略4 抢占
在Go中,⼀个goroutine最多占⽤CPU 10ms,防⽌其他goroutine被饿死

调度策略5 全局队列
当M执⾏work stealing从其他P偷不到G时,它可以从全局G队列获取G。

以上就是详解golang中的gmp模型的详细内容,更多关于详解golang中的gmp模型的资料请关注脚本之家其它相关文章!

相关文章

  • Go单体服务开发最佳实践总结

    Go单体服务开发最佳实践总结

    这篇文章主要介绍了Go单体服务开发最佳实践,通过本文详细跟大家分享一下如何使用 go-zero 快速开发一个有多个模块的单体服务,需要的朋友可以参考下
    2022-04-04
  • Go语言HTTP请求流式写入body的示例代码

    Go语言HTTP请求流式写入body的示例代码

    这篇文章主要介绍了Go语言HTTP请求流式写入body,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-06-06
  • Golang使用Decimal库避免运算中精度损失详细步骤

    Golang使用Decimal库避免运算中精度损失详细步骤

    decimal是为了解决Golang中浮点数计算时精度丢失问题而生的一个库,使用decimal库我们可以避免在go中使用浮点数出现精度丢失的问题,下面这篇文章主要给大家介绍了关于Golang使用Decimal库避免运算中精度损失的相关资料,需要的朋友可以参考下
    2023-06-06
  • Go Java算法之Excel表列名称示例详解

    Go Java算法之Excel表列名称示例详解

    这篇文章主要为大家介绍了Go Java算法之Excel表列名称示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • 详解Golang中链表的创建和读取

    详解Golang中链表的创建和读取

    这篇文章主要为大家详细介绍了Golang中链表的创建和读取的相关知识,文中的示例代码讲解详细,具有一定的借鉴价值,感兴趣的小伙伴可以跟随小编一起了解下
    2023-12-12
  • 关于Golang变量初始化/类型推断/短声明的问题

    关于Golang变量初始化/类型推断/短声明的问题

    这篇文章主要介绍了关于Golang变量初始化/类型推断/短声明的问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-02-02
  • go zero微服务实战系服务拆分

    go zero微服务实战系服务拆分

    这篇文章主要为大家介绍了go zero微服务实战系服务拆分的示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • GOLANG版的冒泡排序和快速排序分享

    GOLANG版的冒泡排序和快速排序分享

    这篇文章主要介绍了GOLANG版的冒泡排序和快速排序分享,需要的朋友可以参考下
    2015-03-03
  • Go并发编程结构体多字段原子操作示例详解

    Go并发编程结构体多字段原子操作示例详解

    这篇文章主要为大家介绍了Go并发编程结构体多字段原子操作示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-12-12
  • Golang使用原生http实现中间件的代码详解

    Golang使用原生http实现中间件的代码详解

    中间件(middleware):常被用来做认证校验、审计等,家常用的Iris、Gin等web框架,都包含了中间件逻辑,但有时我们引入该框架显得较为繁重,本文将介绍通过golang原生http来实现中间件操作,需要的朋友可以参考下
    2024-05-05

最新评论