游戏电视苹果数码历史美丽
投稿投诉
美丽时装
彩妆资讯
历史明星
乐活安卓
数码常识
驾车健康
苹果问答
网络发型
电视车载
室内电影
游戏科学
音乐整形

Golang的调度模型

  Go有四大核心模块,基本全部体现在runtime,有调度系统、GC、goroutine、channel,那么深入理解其中的精髓可以帮助我们理解Go这一门语言!1、Go调度模型发展历史单线程调度器(0。x版本)只包含40多行代码;程序中只能存在一个活跃线程,由GM模型组成;多线程调度器(1。0版本)允许运行多线程的程序;全局锁导致竞争严重;任务窃取调度器(1。1版本)引入了处理器P,构成了目前的GMP模型;在处理器P的基础上实现了基于工作窃取的调度器;在某些情况下,Goroutine不会让出线程,进而造成饥饿问题;(单个p空转)时间过长的垃圾回收(Stoptheworld,STW)会导致程序长时间无法工作;抢占式调度器(1。2版本至今)基于协作的抢占式调度器1。21。13通过编译器在函数调用时插入抢占检查指令,在函数调用时检查当前Goroutine是否发起了抢占请求,实现基于协作的抢占式调度;Goroutine可能会因为垃圾回收和循环长时间占用资源导致程序暂停;基于信号的抢占式调度器1。14至今实现基于信号的真抢占式调度;垃圾回收在扫描栈时会触发抢占调度;抢占的时间点不够多,还不能覆盖全部的边缘情况;非均匀存储访问调度器提案对运行时的各种资源进行分区;实现非常复杂,到今天还没有提上日程;
  上面说到的1。3版本以前历史,其实都是go的非发行版本,所以我们关注与的是go的发行版本,也就是go的gpm模型!G:Goroutine,即我们在Go程序中使用go关键字创建的执行体;M:Machine,或workerthread,即传统意义上进程的线程;P:Processor,即一种人为抽象的、用于执行Go代码被要求局部资源。只有当M与一个P关联后才能执行Go代码。除非M发生阻塞或在进行系统调用时间过长时,没有与之关联的P。
  参考:调度系统设计精要2、GM模型
  P的作用不光光是队列这种抽象,如果理解为队列,那么它的地位只是M的一个子集,GM模型,我们核心关注的是G和M,这也是一般线程池的模型!目标就是高效的调度G在M上!
  下面是我用Go语言简单写的一个调度器,大家可以看看设计思路,以及存在的问题!packagemainimport(fmtgo。uber。orgatomicosossignaltime)funcmain(){sig:make(chanos。Signal,0)监听程序的信号signal。Notify(sig,os。Interrupt,os。Kill)down:make(chanstruct{},0)程序down机信号threadNum:2运行的线程taskQueue:make(chanfunc(),120)任务队列大小限制addTask:func(foofunc()){select{casedown:casetaskQueuefoo:}}schedule:func(){forx:0;xthreadNum;x{gofunc(){for{select{casedown:returncasefoo:taskQueue:foo()}}}()}}schedule()启动调度器count:atomic。Int64{}forx:0;x1;x{addTask(func(){添加一个任务,循环的塞入任务for{addTask(func(){count。Add(1)})}})}等待程序结束select{casesig:close(down)fmt。Println(程序退出,execctrlc)casetime。After(time。Second):close(down)fmt。Printf(调用量:v,count。Load())return}}1、现象
  1、测试条件,调度器只启动两个线程,然后一个线程主要是负责循环的添加任务,一个线程循环的去执行任务gotoolgit:(master)binapp调用量:5078714gotoolgit:(master)binapp调用量:5043506
  2、测试条件,调度器启动三个线程,然后两个线程去执行任务,一个添加任务gotoolgit:(master)binapp调用量:4333959gotoolgit:(master)binapp调用量:4359804
  3、继续测试,启动十个线程,一个添加任务,九个执行任务gotoolgit:(master)binapp调用量:1663691gotoolgit:(master)binapp调用量:1692096
  4、我们添加一些阻塞的任务addTask(func(){count。Inc()time。Sleep(time。Second2)})
  执行可以看到完全不可用gotoolgit:(master)binapp调用量:92、问题
  1、可以看到随着M的不断的增加,可以发现执行任务的数量也不断的减少,原因是什么呢?有兴趣的同学可以加一个pprof可以看看,其实大量的在等待锁的过程!
  2、如果我的M运行了类似于Sleep操作的方法如何解决了,我的调度器还能支撑这个量级的调度吗?
  关于pprof如何使用:在代码头部加一个这个代码:file,err:os。OpenFile(Usersfanhaodonggocodegotoolmainprof。pporf,os。OCREATEos。OTRUNCos。OWRONLY,0644)iferr!nil{panic(err)}iferr:pprof。StartCPUProfile(file);err!nil{panic(err)}deferpprof。StopCPUProfile()
  我们查看一下gotoolpprofmainprof。pporfShowingtop10nodesoutof36flatflatsumcumcum2。45s63。8063。802。45s63。80runtime。usleep0。40s10。4274。220。40s10。42runtime。pthreadcondwait0。28s7。2981。510。28s7。29runtime。(waitq)。dequeueSudoG0。25s6。5188。020。25s6。51runtime。pthreadcondsignal0。17s4。4392。450。94s24。48main。main。func2。10。10s2。6095。050。10s2。60runtime。procyield0。07s1。8296。880。07s1。82runtime。(waitq)。dequeue0。03s0。7897。660。03s0。78runtime。madvise0。02s0。5298。180。99s25。78main。main。func30。02s0。5298。701。72s44。79runtime。selectgo
  可以看到真正执行代码的时间只有0。17s0。02s其他时间都被阻塞掉了!3、GPM模型1、GM模型问题
  1、GM模型中的所有G都是放入到一个queue,那么导致所有的M取执行任务时都会去竞争锁,我们插入G也会去竞争锁,所以解决这种问题一般就是减少对单一资源的竞争,那就是桶化,其实就是每个线程都分配一个队列
  2、GM模型中没有任务状态,只有runnable,假如任务遇到阻塞,完全可以把任务挂起再唤醒运行队列去存放所有的可以运行的任务,runnable所有线程执行运行中的任务,running运行中的任务被阻塞的任务,需要放弃运行权利,挂起到等待队列,blocking2、GPM如何优化GM的(核心)1、引入P
  这里其实会遇到一个问题,假如要分配很多个线程,那么此时随着线程的增加,也会造成队列的增加,其实也会造成调度器的压力,因为它需要遍历全部线程的队列去分配任务以及后续会讲到的窃取任务!
  因为我们知道CPU的最大并行度其实取决于CPU的核数,也就是我们没必要为每个线程都去分配一个队列,因为就算是给他们分配了,他们自己去那执行调度,其实也会出现大量阻塞,原因就是CPU调度不过来这些线程!
  Go里面是只分配了CPU个数的队列,这里就是P这个概念,你可以理解为P其实是真正的资源分配器,M很轻只是执行程序,所有的资源内存都维护在P上!M只有绑定P才能执行任务(强制的)!
  这样做的好处:如果线程很轻,那么销毁和创建会变得很简单,也就是后面会讲到的内核阻塞调用创建线程如果全部资源分配在固定数量的P上,那么可以充分利用CPU并行度2、GPM调度器
  1、首先调度程序其实就是调度不同状态的任务,go里面为Go标记了不同的状态,其实大概就是分为:runnable,running,block等,所以如何充分调度不同状态的G成了问题,那么关于阻塞的G如何解决,其实可以很好的解决G调度的问题!在channel上发送和接收网络IO操作阻塞的系统调用使用定时器或者Sleep使用互斥锁sync。Mutex
  上面这些情况其实就分为:用户态阻塞内核态阻塞但是不会挂起当前线程内核态阻塞会挂起当前线程
  2、用户态阻塞,一般Go里面依靠gopark函数去实现,大体的代码逻辑基本上和go的调度绑定死了
  源码在:https:golang。orgsrcruntimeproc。gofuncgopark(unlockffunc(g,unsafe。Pointer)bool,lockunsafe。Pointer,reasonwaitReason,traceEvbyte,traceskipint){ifreason!waitReasonSleep{checkTimeouts()timeoutsmayexpirewhiletwogoroutineskeeptheschedulerbusy}mp:acquirem()gp:mp。curgstatus:readgstatus(gp)ifstatus!Grunningstatus!Gscanrunning{throw(gopark:badgstatus)}mp。waitlocklockmp。waitunlockfunlockfgp。waitreasonreasonmp。waittraceevtraceEvmp。waittraceskiptraceskipreleasem(mp)cantdoanythingthatmightmovetheGbetweenMshere。mcall(parkm)}parkcontinuationong0。funcparkm(gpg){g:getg()iftrace。enabled{traceGoPark(g。m。waittraceev,g。m。waittraceskip)}casgstatus(gp,Grunning,Gwaiting)dropg()iffn:g。m。waitunlockf;fn!nil{ok:fn(gp,g。m。waitlock)g。m。waitunlockfnilg。m。waitlocknilif!ok{iftrace。enabled{traceGoUnpark(gp,2)}casgstatus(gp,Gwaiting,Grunnable)execute(gp,true)Scheduleitback,neverreturns。}}schedule()调度程序}他会标记当前的g的状态从runningwaiting状态然后开始执行调度:schedule()方法首先就是看看有没有其他特殊情况:比如GC或者trace其次就是可能先去获取全局队列的(代码里写的偶尔,根据随机性去执行的)获取当前g绑定的p队列里获取g从其他地方获取可以运行的g(优先级是:从本地队列全局队列网络轮巡器窃取其他P的G,具体可以看findrunnable方法)最后执行那个唤醒的G最后就是unpark,就是将当前goroutine置于等待状态并解锁锁。可以通过调用goready方法使goroutine再次运行。
  3、其实对于netpool这种nio模型,其实内核调用是非阻塞的,所以go开辟了一个网络轮训器队列,来存放这些被阻塞的g,等待内核被唤醒!那么什么时候会被唤醒了,其实就是需要等待调度器去调度了!n,err:syscall。Read(fd。Sysfd,p)iferr!nil{n0iferrsyscall。EAGAINfd。pd。pollable(){iferrfd。pd。waitRead(fd。isFile);errnil{这里会等待读,其实就是挂起了当前g,也就是主动让出了mcontinue}}}
  4、如果是内核态阻塞了(内核态阻塞一般都会将线程挂起,线程需要等待被唤醒),我们此时P只能放弃此线程的权利,然后再找一个新的线程去运行P!
  关于着新线程:找有没有idle的线程,没有就会创建一个新的线程!
  关于当内核被唤醒后的操作:因为GPM模型所以需要找到个P绑定,所以G会去尝试找一个可用的P,如果没有可用的P,G会标记为runnable放到全局队列中!
  关于内核唤醒后G如何执行的代码我没有找到,不好意思,逻辑没有看太清晰!其实疑问点:如何找到可用的P,所以固定数量的P的好处就是查询时间比较可控!为何不找到上一次绑定的P呢?为何切换上下文了!
  5、其实了解上面大致其实就了解了Go的基本调度模型G运行的优先级是啥P和M啥时候会接触绑定P啥时候会窃取G
  答案文章里慢慢品味!3、如何防止G长时间占用P
  如果某个G执行时间过长,其他的G如何才能被正常的调度?这便涉及到有关调度的两个理念:协作式调度与抢占式调度。协作式和抢占式这两个理念解释起来很简单:协作式调度依靠被调度方主动弃权;抢占式调度则依靠调度器强制将被调度方被动中断。1、空转代码
  例如下面的代码,我本地的版本是go1。13。5packagemainimport(fmtruntime)funcmain(){gofunc(){for{{}}}()runtime。Gosched()表示主线程让出当前线程,可以理解为先让go执行fmt。Println(close)}2、存在的问题
  执行:GOMAXPROCS1配置全局只能有一个Pgotoolgit:(master)GODEBUGschedtrace1GOMAXPROCS1binappSCHED0ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue0〔2〕SCHED1ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED2ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED4ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED7ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED13ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕
  可以看到main函数无法执行!也就是那个go空转抢占了整个程序
  备注:GODEBUGschedtrace1表示1ms打印一次SCHED:调试信息输出标志字符串,代表本行是goroutinescheduler的输出;1ms:即从程序启动到输出这行日志的时间;gomaxprocs:P的数量;idleprocs:处于idle状态的P的数量;通过gomaxprocs和idleprocs的差值,我们就可知道执行go代码的P的数量;threads:osthreads的数量,包含scheduler使用的m数量,加上runtime自用的类似sysmon这样的thread的数量;spinningthreads:处于自旋状态的osthread数量;idlethread:处于idle状态的osthread的数量;runqueue1:goscheduler全局队列中G的数量;〔34010〕:分别为4个P的localqueue中的G的数量。3、升级1。14版本解决
  但是假如我换为用1。14版本执行,有兴趣的话可以使用我的docker镜像,直接可以拉取:fanhaodonggolang:1。15。11和fanhaodonggolang:1。13。5〔root647af84b3319code〕GODEBUGschedtrace1GOMAXPROCS1binappSCHED0ms:gomaxprocs1idleprocs0threads2spinningthreads0idlethreads0runqueue0〔0〕SCHED1ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue0〔0〕SCHED2ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue0〔0〕SCHED3ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED4ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED5ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED6ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED7ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED8ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED10ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕SCHED13ms:gomaxprocs1idleprocs0threads3spinningthreads0idlethreads1runqueue1〔1〕close4、关于G上内存分配的问题
  首先我们知道GMP,G可能和M也可能和P解除绑定,那么关于数据变量放在哪哇!其实这个就是逃逸分析!1、什么情况下会逃逸packagemaintypeDemostruct{Mainstring}funcmain(){gofunc(){demo:Demo{}gofunc(){test(demo)}()}()}functest(demoDemo){}
  输出可以看到其实没有发生逃逸,那是因为demo被拷贝它自己的栈空间内〔root647af84b3319code〕gobuildgcflagsNlmobinappdenomain2。gocommandlineargumentsdenomain2。go:13:11:demodoesnotescapedenomain2。go:6:5:funcliteralescapestoheapdenomain2。go:8:6:funcliteralescapestoheap
  备注:
  gcflagsNlm其中N禁用优化l禁止内联优化,m打印逃逸信息
  那么继续改成这个packagemaintypeDemostruct{Mainstring}funcmain(){gofunc(){demo:Demo{}gofunc(){test(demo)}()}()}functest(demoDemo){}
  可以看到发现demo对象其实被逃逸到了堆上!这就是不会出现类似于G如果被别的M执行,其实不会出现内存分配位置的问题!〔root647af84b3319code〕gobuildgcflagsNlmobinappdenomain2。gocommandlineargumentsdenomain2。go:13:11:demodoesnotescapedenomain2。go:7:3:movedtoheap:demodenomain2。go:6:5:funcliteralescapestoheapdenomain2。go:8:6:funcliteralescapestoheap2、for循环中申明g内存引用问题
  所以可以看到demo其实是copy到了堆上!这就是g逃逸的问题,和for循环一样的funcmain(){forx:0;x10;x{gofunc(){fmt。Println(x)}()}}
  执行可以发现,其实x已经逃逸到了堆上,所以你所有的g都引用的一个对象,如何解决了gotoolgit:(master)gobuildgcflagslmobinappdenomain2。gocommandlineargumentsdenomain2。go:10:6:movedtoheap:xdenomain2。go:11:6:funcliteralescapestoheapdenomain2。go:12:15:main。func1。。。argumentdoesnotescapedenomain2。go:12:15:xescapestoheapdenomain2。go:16:11:testdemodoesnotescape
  如何解决了,其实很简单直接copy一份数据forx:0;x10;x{x:xclonegofunc(){fmt。Println(x)}()}通过参数传递(推荐)forx:0;x10;x{gofunc(xint){fmt。Println(x)}(x)}参考文章
  也谈goroutine调度器
  图解Go运行时调度器
  Go语言回顾:从Go1。0到Go1。13
  Go语言原本
  调度系统设计精要
  ScalableGoSchedulerDesignDoc

开惯了汉兰达,换开朋友的昂科威PLUS,终于明白日系美系的差我们经常听人说国人买车喜欢大的,这话一点不假,只要预算充足,尺寸是越大越好。并不是国人不懂车,而是国内经济这几年飞速发展,国民消费能力不断攀升,从入门将就到舒适豪华,消费者的多……65瓦快充配骁龙865,realmeX50Pro玩家版269万铎黄阳发自凹非寺鹅板凳公众号ebandeng上半年的旗舰机发布热潮刚刚结束,最近各个厂家又像是商量好了一样,开始扎堆发布各自的次旗舰机型。今天,realme……试试11引擎的精度,赛睿RIVAL106高低速移动测试赛睿TrueMove引擎最低端的是TrueMove1,用在RIVAL106上。作为定价200内的鼠标,RIVAL106的关键卖点也在这里,那么它的精准度是否比其它鼠标更加出色?……年销10万件箱包,外贸企业借助拼多多自创品牌从零到10万过去一年间,全球疫情处于复杂多变的态势,叠加不间断的经贸摩擦,中国外贸的不确定因素高企,一度面临严峻的考验。值此局面下,中国着力打造以国内大循环为主体、国内国际双循环相互……硬核测评小米机械键盘CHERRY轴版的大键手感去年小米推出一款300元内的樱桃轴键盘,不管是设计风格,还是从它的命名和型号来看,它都和我们熟悉的ikbc那种OEM机械键盘不同,最不习惯的还得是它连个像样好记的名字都没,产品……如果你在闲鱼被骗,请按照以下流程维权上次发了那个闲鱼被骗追回钱的视频视频加载中。。。收获了很多私信真的没有办法每一个人都回复,所以这里统一写一下首先说一下我碰到的是什么情况我碰到的是……吃鱼的讲究,烧鱼不碎的要领一、吃鱼的讲究买回的鱼如果新鲜,最好清蒸,这样营养破坏少,且可保持鲜味;如果鱼的新鲜度稍差,则宜红烧。在烹调的时候,加适量醋,可保护维生素C的稳定性,而且还可促进钙的溶出……单页面的黄金时代什么是单页面应用单页面应用的英文名称是SPA(singlepageapplication),和传统的多页面应用不同,它只有一个页面,页面的内容变化都是通过js来进行动态修……厨子qdc旗舰十动铁耳塞VX评测qdc把整个市场给做开了,这是一句来自友商对于qdc品牌的评价。前阵子我和朋友聊到有关定制耳机的话题,我们不约而同的想到了近年来HIFI市场大热的国产品牌qdc。确实,qdc是……郭广昌,复星集团创始人,要做中国的巴菲特1967年,郭广昌出生在浙江东阳的农村,寒窗苦读多年,1985年考上了复旦大学哲学系。由于他在大学的表现优异,所以获得留校任教的宝贵机会,郭广昌也因此作了一段时间的大学老师。……吉利甘当富士康,集度而来的百度造车这事儿能成吗?百度要造车了,这个消息在今年1月份由百度官方宣布,虽然此前百度将涉足造车业务的一些消息已被释放,但是官宣之后,还是给全球车企带来了不小的震动,因为百度不单是自己造车,而是拉上了……关羽号称忠勇无敌,投降曹操竟是为保护如花似玉的嫂子关羽投降曹操这事,还得从建安五年说起。建安五年,不甘心做傀儡皇帝的汉献帝刘协,秘密颁给董承一个衣带诏,命其召集忠勇之士诛杀曹操,结局汉室的危机。刘备名列其中。后来,衣带诏东窗事……
光之未来,一加9R这次的一加直接找来了迪迦做代言人,真是吸引了一群00后的当代青年〔来看我〕〔奸笑〕汇总一下安卓之光一加9R的参数:6。55英寸三星E3发光材料SuperAMOLED……应急必备!比德生更全能的六边形战士,手摇发电,720天超长待最近,各地频发暴雨。全球变暖,导致极端天气越来越多,不说国内,在遥远的北极圈也正在发生令科学家诧异的事情,格陵兰冰盖温度上升到0以上,三天内下了史无前例的70亿吨雨水,是大半个……任正非很欣慰!雷军终于认清事实,网友国产不被卡脖子指日可待任正非一直在倡导的科学是什么样的?我们用4件事来看看。事件一2018年7月26日,包括任正非在内的华为最高管理层,在深圳华为坂田基地的大门口等了十几分钟,他们等的是一个叫……GalaxyNote20换装纯平屏幕仅限基础版本本周早些时候有传言称,三星可能会放弃GalaxyNote20的曲面屏设计,不过,目前尚不清楚这种新的纯平屏幕设计是仅限基础机型采用,还是包括GalaxyNote20Note20……精彩预告2021(首届)全球数智营销峰会数据驱动智赢未来2021年初,云徙科技发起了一场名为数智中国行的数字化全国巡展活动,旨在通过全国巡展的形式,将云徙5年来沉淀的基于中台的数字化营销实战经验与案例,在全国范围内做分享,从而响应国……这份榜单发布后,谁还看不起原神?SensorTower商店情报平台发布了2021年1月手游产品收入和下载量数据,前TOP30榜单如下:1月米哈游《原神》海外移动端收入达到1。14亿美元,这已经是该游戏连……十来万的车就别看合资品牌了!吉利帝豪S不香吗?标配1。4T发在国内汽车市场,合资二字是带有信仰加成的,即使国产车做得再好,在合资党眼中也是低档货,其实理性的想一想,十来万的合资车能有什么面子?十来万又能得到什么?这个价格还是国产车更香,……安兔兔公布手机性能排行榜,iQOO5系列包揽前两名一款手机好不好,很多人都会看榜单,尤其是安兔兔的榜单,会经常给出一些手机的榜单,诸如性能,性价比之类的,还是有一些参考价值的。安兔兔今天给出了最新的安卓手机性能排行榜,当然这个……禁用系统的广告弹窗,我有妙招你的电脑是不是每天一开机就会弹出很多广告,每次都费尽心思去关掉那些弹窗。而且有时候那些弹窗的内容还是不堪入目,尝试了各种各样的方法还是有弹窗广告。今天就教大家几招,可以去……70平纯白小家,衣帽间洗衣区电竞区,想要的通通都有本案屋主是一对同在地铁工作的夫妻,有着精致且新潮的审美观,热衷电竞,追究细节,他们不仅对住宅设计有自己的要求,在对家电选择上,也有自己的一套标准,我们从实景中来看。房屋坐……java基础Spring(一)1。3。1Spring的概述1。3。1。1什么是SpringSpring:SEEE开发的一站式框架。一站式框架:有EE开发的每一层解决方案。WEB层:S……重磅!缺箱爆仓!多家船公司采取了这些新举措由于今年受疫情原因,现如今从中国出口的柜量慢慢在增加,国外进口到中国的柜量逐渐减少,导致部分船公司出现了国外返程空船、亏载的情况。1hr部分船公司已发出缺箱调整通知:赫伯……
友情链接:易事利快生活快传网聚热点七猫云快好知快百科中准网快好找文好找中准网快软网