Golang—— GMP模型发展(一)

一、前言

Golang语言的关键点之一:协程;相对于线程而言,协程是一个更轻量级的线程。在讲到协程的时候,不可避免的要讲到GMP模型,而GMP模型的前身是GM模型。

二、GM与GMP

Go1.1版本(2012年)之前用的都是GM模型,但是由于GM模型性能不好,之后官方对调度器进行了改进,演化为现在的GMP模型。

GM模型

GM模型中的G全称为Goroutine协程,M全称为Machine内核级线程,其调度过程如下:

M(内核级线程)从加锁的Goroutine队列中获取G(协程)执行,如果G在运行过程中创建了新的G,那么新的G也会被放在全局队列中。

基于以上设计存在两个不足之处:

  • 调度,返回G都需要获取队列锁,形成了激烈竞争。
  • M转移G没有把资源最大化利用。比如当M1在执行G1时,M1创建了G2,为了继续执行G1,则需要把G2交给M2执行,因为G1和G2是相关的,而寄存器中会保存G1的信息,因此G2最好是放在M1上执行,而不是其他M。

GMP模型

GMP模型是在GM模型基础上增加了一个P层。

G:goroutine,协程

M:machine,内核级线程

P:processor,协程运行所需要的资源。

全局队列:当P中的本地队列有G溢出时,会被放在全局队列中;

P的本地队列:P内置的G队列存在数量限制(≤256个)。 此处有

  • 当队列P1中的G1在运行过程中创建G2,G2会优先被放在P1本地队列中,如果P1本地队列满了,则会把P1队列中一半的G移动到全局队列中;
  • 如果P的本地队列为空,那么他会先到全局队列中获取G,如果全局队列中也没有G,则尝试从其他线程绑定的P中偷取一半的G

P与M的数量是不能无限扩增的:

P的数量:由启动时环境变量($GOMAXPROCS) 或者由runtime的方法GOMACPROCS()决定。

M的数量:go程序启动时,会设置M的最大数量,默认为10000。但是根据内核很难创建出这麽多的线程,因此在默认情况下的最大数量取决于内核;也可以调用runtime/debug中的SetMaxThreads函数,手动设置M的最大数量。

P与M的创建时机:

P:在确定P的最大数量(n)后,运行时系统会根据这个数量创建n个p,即运行时创建。

M:内核级线程的初始化是由内核管理的,当没有足够的M来关联P 并运行其中可运行的G时,会请求创建新的M。

比如:M在运行G1时被阻塞住了,此时需要新的M去绑定P,如果没有在休眠的M则需要创建新的M。

 G在GMP模型中的执行过程

  1.  调用go func()创建一个goroutine;
  2. 新创建的G有限保存在P的本地队列中,如果本地队列满了,则会保存在全局队列中;
  3. M需要在P的本地队列中获取一个可用G,如果P的队列为空则会先从全局队列获取G,如果全局队列也为空则去其他P对队列中偷取G放在自己的P中;
  4. G将相关参数传输给M,为M执行G做准备;
  5. 当M执行某个P时,如果发生系统调用产生导致M会阻塞;如果当前P队列中有一些G,runtime会将线程M和P分离,然后再获取空闲的线程或者创建一个新的M(内核级线程)来服务这个P,阻塞调用完成后G被销毁,将值返回;
  6. 销毁G,将执行结果返回;
  7. 当M系统调用结束时,这个M会尝试获取一个空闲的P执行,如果获取不到P,则这个M变成休眠状态,加入到空闲线程中。

三、两者区别

  • GMP相对于GM做的优化点有三个:

  1. 每个P都有自己的本地队列,而不是所有的G操作都经过全局队列,这样锁的竞争会少很多。(GM模型性能开销最大的就是锁竞争)

  2. P的本地队列平衡上实现了Work Stealing算法。如果P的本地队列为空则会从全局队列或者其他P队列中窃取可运行的G来运行(通常偷一半),减少空转,提高资源利用率。

  3. hand off机制。当M0线程因为G1进行系统调用阻塞时,M释放绑定的P,把P转移到其他空闲的M执行,同样提高了资源利用率。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: GMP (GNU Multiple Precision Arithmetic Library) 是一个开源的多精度数学库,支持高精度的整数和浮点数运算。Golang 有一个内置的多精度数学包(math/big),它使用了 GMP 库来实现高精度运算。 Golang 的 math/big 包提供了一些基本的数学运算,包括加法、减法、乘法、除法和模运算。它还提供了一些高级的函数,如幂运算、对数、三角函数和取整函数。 Golang 的 math/big 包非常适用于处理高精度的数学运算,如货币计算、加密、科学计算等。它的精度可以达到任意精度,而且运算速度也相当快。 总的来说,GolangGMP 模型是一个非常优秀的多精度数学库,适用于各种高精度数学运算的场景。 ### 回答2: Golang中的gmp模型指的是Goroutine、Mutex、和Channel三个概念的组合使用。 首先,Goroutine是轻量级的线程,可以通过go关键字启动。Goroutine利用multiplexer调度器实现并发执行。它不仅启动速度快,而且占用的内存资源较少,可以高效地处理大量任务。 接下来,Mutex是一种互斥量,用于保护共享资源的访问。在多个Goroutine并发执行的情况下,Mutex可以保证在同一时间只有一个Goroutine能够访问共享资源,避免数据竞争的问题。 最后,Channel是一种用于在Goroutine之间通信的管道。它可以实现Goroutine之间的同步和数据传递。Channel提供了发送和接收操作,可以在不同的Goroutine之间发送和接收数据,确保数据的安全和一致性。通过Channel,Goroutine可以并发地进行通信和协作,实现更高效的并发编程。 Golanggmp模型的优点是简洁而高效。Goroutine提供了轻量级的并发模型,使得并发编程变得简单而高效。Mutex在保护共享资源的同时,避免了死锁的问题。Channel则提供了通信和同步的机制,保证了数据的安全和一致性。 此外,通过使用gmp模型,我们可以充分利用多核处理器的计算能力,提高程序的并发性能。Goroutine的调度器能够智能地利用多核处理器的资源,实现任务的并发执行,提高程序的运行效率。 总结来说,Golanggmp模型提供了一种简洁而高效的并发编程模型,使得我们能够轻松地实现并发编程,并充分利用多核处理器的计算能力。通过合理地使用Goroutine、Mutex和Channel,我们能够编写出高效且可靠的并发程序。 ### 回答3: gmpgolang中的一种高性能并发模型,它是一种轻量级的基于goroutine的消息传递模型。与传统的共享内存模型相比,gmp模型更加安全和简单。 gmp模型的核心思想是通过goroutine之间的通信来实现任务的执行和数据的传递。在gmp模型中,每个任务都可以作为一个goroutine来执行,并与其他goroutine进行消息的传递。这种消息传递是通过通道(channel)来实现的,通道是goroutine之间的连接,用于传递数据和控制消息的通信方式。通过通道,可以实现任务之间的同步和协作,有效地避免了传统共享内存模型中的数据竞争和死锁等问题。 在gmp模型中,每个goroutine都可以独立运行,无需关心其他goroutine的存在和状态。这样可以极大地简化程序的编写和维护,提高了代码的可读性和可维护性。此外,gmp模型还提供了丰富的并发原语,如锁、等待组、原子操作等,以支持更复杂的并发控制和同步操作。 相比于其他并发模型gmp模型的优势在于其简单性和易用性。并且由于golang本身对并发模型的良好支持,因此在实际使用中,gmp模型可以帮助开发者更轻松地编写高效且具有良好并发性能的程序。 然而,gmp模型也存在一些局限性。由于通道是goroutine之间的连接,因此通道的创建和关闭操作等极易受到限制。另外,在某些情况下,使用共享内存模型可能更加高效,因为通道的消息传递机制可能会引入一定的开销。因此,在选择并发模型时,开发者需要根据具体的应用场景和性能需求进行权衡。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Mr.Qubb

你的鼓励是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值