Golang的调度模型

Python015

Golang的调度模型,第1张

Go有四大核心模块,基本全部体现在runtime,有调度系统、GC、goroutine、channel,那么深入理解其中的精髓可以帮助我们理解Go这一门语言!

参考: 调度系统设计精要

下面是我用Go语言简单写的一个调度器,大家可以看看设计思路,以及存在的问题!

1、测试条件,调度器只启动两个线程,然后一个线程主要是负责循环的添加任务,一个线程循环的去执行任务

2、测试条件,调度器启动三个线程,然后两个线程去执行任务,一个添加任务

3、继续测试,启动十个线程,一个添加任务,九个执行任务

4、我们添加一些阻塞的任务

执行可以看到完全不可用

1、 可以看到随着M的不断的增加,可以发现执行任务的数量也不断的减少,原因是什么呢?有兴趣的同学可以加一个pprof可以看看,其实大量的在等待锁的过程!

2、如果我的M运行了类似于Sleep操作的方法如何解决了,我的调度器还能支撑这个量级的调度吗?

关于pprof如何使用:在代码头部加一个这个代码:

我们查看一下 go tool pprof main/prof.pporf

可以看到真正执行代码的时间只有 0.17s + 0.02s 其他时间都被阻塞掉了!

1、GM模型中的所有G都是放入到一个queue,那么导致所有的M取执行任务时都会去竞争锁,我们插入G也会去竞争锁,所以解决这种问题一般就是减少对单一资源的竞争,那就是桶化,其实就是每个线程都分配一个队列

2、GM模型中没有任务状态,只有runnable,假如任务遇到阻塞,完全可以把任务挂起再唤醒

这里其实会遇到一个问题,假如要分配很多个线程,那么此时随着线程的增加,也会造成队列的增加,其实也会造成调度器的压力,因为它需要遍历全部线程的队列去分配任务以及后续会讲到的窃取任务!

因为我们知道CPU的最大并行度其实取决于CPU的核数,也就是我们没必要为每个线程都去分配一个队列,因为就算是给他们分配了,他们自己去那执行调度,其实也会出现大量阻塞,原因就是CPU调度不过来这些线程!

Go里面是只分配了CPU个数的队列,这里就是P这个概念,你可以理解为P其实是真正的资源分配器,M很轻只是执行程序,所有的资源内存都维护在P上!M只有绑定P才能执行任务(强制的)!

这样做的好处:

1、首先调度程序其实就是调度不同状态的任务,go里面为Go标记了不同的状态,其实大概就是分为:runnable,running,block等,所以如何充分调度不同状态的G成了问题,那么关于阻塞的G如何解决,其实可以很好的解决G调度的问题!

上面这些情况其实就分为:

2、用户态阻塞,一般Go里面依靠 gopark 函数去实现,大体的代码逻辑基本上和go的调度绑定死了

源码在:https://golang.org/src/runtime/proc.go

3、其实对于netpool 这种nio模型,其实内核调用是非阻塞的,所以go开辟了一个网络轮训器队列,来存放这些被阻塞的g,等待内核被唤醒!那么什么时候会被唤醒了,其实就是需要等待调度器去调度了!

4、如果是内核态阻塞了(内核态阻塞一般都会将线程挂起,线程需要等待被唤醒),我们此时P只能放弃此线程的权利,然后再找一个新的线程去运行P!

关于着新线程:找有没有idle的线程,没有就会创建一个新的线程!

关于当内核被唤醒后的操作:因为GPM模型所以需要找到个P绑定,所以G会去尝试找一个可用的P,如果没有可用的P,G会标记为runnable放到全局队列中!

5、其实了解上面大致其实就了解了Go的基本调度模型

答案文章里慢慢品味!

如果某个 G 执行时间过长,其他的 G 如何才能被正常的调度? 这便涉及到有关调度的两个理念:协作式调度与抢占式调度。协作式和抢占式这两个理念解释起来很简单: 协作式调度依靠被调度方主动弃权;抢占式调度则依靠调度器强制将被调度方被动中断。

例如下面的代码,我本地的版本是 go1.13.5

执行: GOMAXPROCS=1 配置全局只能有一个P

可以看到main函数无法执行!也就是那个go 空转抢占了整个程序

备注:

但是假如我换为用 1.14+版本执行,有兴趣的话可以使用我的docker镜像,直接可以拉取: fanhaodong/golang:1.15.11 和 fanhaodong/golang:1.13.5

首先我们知道G/M/P,G可能和M也可能和P解除绑定,那么关于数据变量放在哪哇!其实这个就是逃逸分析!

输出可以看到其实没有发生逃逸,那是因为 demo被拷贝它自己的栈空间内

备注:

-gcflags"-N -l -m" 其中 -N禁用优化-l禁止内联优化,-m打印逃逸信息

那么继续改成这个

可以看到发现 demo对象其实被逃逸到了堆上!这就是不会出现类似于G如果被别的M执行,其实不会出现内存分配位置的问题!

所以可以看到demo其实是copy到了堆上!这就是g逃逸的问题,和for循环一样的

执行可以发现,其实x已经逃逸到了堆上,所以你所有的g都引用的一个对象,如何解决了

如何解决了,其实很简单

也谈goroutine调度器

图解Go运行时调度器

Go语言回顾:从Go 1.0到Go 1.13

Go语言原本

调度系统设计精要

Scalable Go Scheduler Design Doc

全栈开发专业致力于培养具有前端,后端,移动端,小程序多端开发能力,具有项目统筹能力,具有部署与运维能力,全局思维管理能力,团队协作能力的全栈开发人才。

为什么学习全栈专业——

全栈专业承载着互联网变革的使命,覆盖前后端的所有技术,就业选择多,就业速度快

为什么全栈能优质就业——

需要具备全局思维能力、部署与运维技术,全栈开发人才能够满足企业需要

全栈人才的发展空间——

扎实的技术功底,超强的思维方式,学生可以掌握项目研发各个环节,优质就业不是梦

全栈开发专业特色——

1.学生平均就业薪资2万元左右

2.集结了全栈开发专业最具优势的技术力量团队,以企业实战项目授课

3.面向行业最新的技术,掌握最前沿的技术知识

4.定期组织学员去阿里巴巴(杭州)等知名互联网公司学习体验

5.实训主任及专业老师为教学及就业工作服务,确保学生优质就业,入学全栈专业的学生日常生活一切问题,均可以与实训主任直接对话解决

全栈开发专业优势——

首先必须明确的是,“全栈”不是“全能”,广泛的 Web 技术才是一名通常意义上的全栈工程师最该专注的核心内容,而此处的 Web 技术,包括网络、前端、后端 MVC、持久化技术等多个层次。

那么成为一名“全栈工程师”你可以收获什么?

掌握快速获得反馈的学习方法,通过联结和类比的方式,构建完整的知识体系。

拓展视野,养成思考习惯,独立决策、交付一套完整的产品。

更多的职场就业机会,全栈工程师本身就拥有多技能、多角色,团队非常欢迎这样的人。

更佳的创业技术角色,有了创意和想法,全栈工程师就可以快速地开发原型,迅速实现第一个版本,并能马上提供优质的宣传推广和用户体验。

全栈开发专业培养目标——

这个世界需要人才,但更需要通晓各个层面知识,能够独立、快速解决问题的人。“全栈工程师”能成为你的职业上升通道上的一个驿站,为你的成功提供一个高度。

全栈开发专业致力于培养

具有前端,后端,移动端,小程序多端开发能力,具有项目统筹能力,具有部署与运维能力,全局思维管理能力,团队协作能力的全栈开发人才。

从就业前景上来讲

全栈学习毕业后可选择进入阿里,京东等知名互联网企业,岗位有:高级全栈开发师,技术总监,全栈产品经理,系统架构师,项目总监,前端工程师,后台开发工程师等。

课程体系——

专业基础阶段:计算机基础、网页设计、程序设计基础、C语言精髓

专业阶段:前端基础、前端进阶、后端编程、全栈开发实战、全栈性能优化基础

专业高级阶段:全栈性能优化进阶、GO语言入门与项目实战、G0语言大数据、项目实战、全栈移动项目实战、高并发高可用优化实战、全栈项目实战与综合复习