对于Golang中GMP模型的理解

1 概念介绍

golang协程实现的核心机制是GMP
G代表goroutine协程,M代表内核态线程,P代表处理器(不等同于CPU核心)。
整个GMP模型从下往上可以按照程序空间分为两部分:运行在内核态的部分、运行在用户态的调度器。
运行在内核态的部分从下往上分别为底层硬件CPU核心、OS级别调度器、多个内核线程M。M是真正运行G的实体,OS调度器负责把内核线程分配到底层硬件的CPU核心上去实际执行。
调度器的功能是把可运行的G分配到可以工作的M上。调度器的组成包括P、P维护的本地G队列、全局G队列。P包含了运行G的资源,如果M想运行G,则必须先获取P。全局G队列维护了当前等待运行的G,P的本地G队列与全局G队列作用类似,不过其最多存256个G,如果新建G时发现队列满,则将处于已满状态的P本地队列中的一半G打乱顺序移动到全局队列。在程序启动时一次性创建所有的P(共GOMAXPROCS个),存放在数组中。
M和P的数量没有绝对的关系,因为如果一个M发生了阻塞,那么P就会去尝试寻找空闲的M,如果没有空闲的,则会创建新的M。

2 调度器策略

2.1 线程复用

避免频繁的创建、销毁线程,而是尽可能复用线程。

work stealing机制

M在获取P、运行G时,首先尝试从P的本地队列获取G,若队列为空,则M也会先去尝试从其他P的本地队列偷一半放到自己P的本地队列。

hand off机制

当M因为G进行系统调用阻塞时,M会释放绑定的P,把P转移给其他空闲的M执行。

2.2 控制并行

可以基于GOMAXPROCS来控制并行程度,比如设置GOMAXPROCS=CPU核心数/2,则最多有一半的CPU核心会并行运行线程。

2.3 抢占原则

一个G一次最多占用CPU10ms时间片,10ms后要重新参与调度,以防止出现G的饥饿现象。这里的抢占式调度实现原理如下。
golang在程序启动的时候,会专门创建一个线程sysmon,用来监控和管理,在内部是一个循环:
1.记录所有P的G任务计数schedtick,schedtick会在每执行一个G任务后递增;
2.如果检查到schedtick一直没有递增,说明这个P一直在执行同一个G任务,如果超过10ms,就在这个G任务的栈信息里面加一个标记;
3.然后这个G任务在执行的时候,如果遇到非内联函数调用,就会检查一次这个标记,然后中断自己,把自己加到队列末尾,执行下一个G;
4.如果没有遇到非内联函数(有时候正常的小函数会被优化成内联函数)调用的话,会一直执行这个G,直到它自己结束,此时如果是个死循环,并且GOMAXPROCS=1的话,程序就永远不会自己停止,且CPU会一直被占用。

3 调度器生命周期

在这里插入图片描述
M0是启动程序后的编号为0的主线程,这个M对应的实例会在全局变量runtime.m0中,不需要在堆上分配,M0负责执行初始化操作和启动第一个G,在之后M0就和其他的M一样了。
G0是每次启动一个M都会第一个创建的gourtine,G0仅负责调度的G,其不指向任何可执行的函数,每个M都会有一个自己的G0。在调度或系统调用时会使用G0的栈空间,全局变量中的G0是 M0的G0。

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: GMP (GNU Multiple Precision Arithmetic Library) 是一个开源的多精度数学库,支持高精度的整数和浮点数运算。Golang 有一个内置的多精度数学包(math/big),它使用了 GMP 库来实现高精度运算。 Golang 的 math/big 包提供了一些基本的数学运算,包括加法、减法、乘法、除法和模运算。它还提供了一些高级的函数,如幂运算、对数、三角函数和取整函数。 Golang 的 math/big 包非常适用于处理高精度的数学运算,如货币计算、加密、科学计算等。它的精度可以达到任意精度,而且运算速度也相当快。 总的来说,GolangGMP 模型是一个非常优秀的多精度数学库,适用于各种高精度数学运算的场景。 ### 回答2: Golanggmp模型指的是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
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值