有了解过协程吗?介绍 goroutine

协程一直来说概念上挺玄学的,个人觉得确实是属于网上误传也非常多一种概念,经常跟用户级线程、绿色线程各种概念混在一起了。很多时候都说是一个东西,很多时候又能说出区别。
留一个自己认知的大概解就差不多了,关键落到会用,跟线程的对比上有足够认知就差不多了。要提到这些概念也没问题,面试别咬死了说就好(我目前感觉这三个确实是一个东西)。
● 不管定义是怎样,一定正确的一些特点:
  ○ 协作式,非抢占式,编程语言决定或者协程自己决定如何做切换调度
  ○ 完全用户态,无内核开销,大多数操作不涉及系统调用
● 跟线程对比
  ○ 实现特点:线程是 os 的软中断实现的,内核决定主要调度,协程一般是自己让渡,很多时候是线程复用,无传统意义上的中断。
  ○ 控制难度:这个是我之前没留意的一个点,感觉协程线程其实用起来都差不多。但协程确实提供了很多机制来降低控制难度,线程需要百分百管理锁和同步,而协程一般都会在语言运行时层面上进一步做抽象来简化并发模型。比如 go 里边的 wg 和 channel。
  ○ 性能:更轻量实现并发。上下文切换、用户态、占用内存(栈空间)

goroutine 是协程在 go 语言里的‘实现’(我觉得实现这种说法比是在 go 里面的‘优化’更好一点,因为协程本身就没有一个确定基础实现。所以其实就我目前的理解协程更像是满足上面特点的一种‘不同于多线程的程序并发实现思想’)
● 基本认知
  ○ 和线程 m 对 n 关系,可以并行
  ○ 基于 GMP 模型,模型可以认知为整体的一个实现,其中 P 是作为调度器的角色,gm 分别代表 goroutine 和 内核线程的抽象。
  ○ 几种调度方式:一种分类是主动、被动、正常、抢占四种情况。这个分类挺好的,列举了所有情况。主动是还没执行完交出执行权,被动是临时阻塞被调度器给调度了,正常调度是结束调度新的,抢占调度是后面版本引入的针对系统调用僵直情况的调度。前面三种都是协作式的符合协程特点。
● 怎样做调度,大概的调度流程和机制,剩下有哪些调度细节
  ○ gmp 模型中,p 主要是本地队列,m 主要是线程抽象,直接指向 g。执行设想就是,p 动态绑定一个 m,然后 m 再通过 p 去拿 g 去执行。
  ○ m 怎么找到下一个正常执行的 g:主要按照本地队列、全局队列、wait 队列、work-stealing 的顺序取(再加上一个每 61 次会优先全局防止饥饿)
○ g 的生命周期,正常流程 idle -> dead -> runnable -> running -> dead,运行的两状态会分别进入一个系统调用状态和 wait 状态(即阻塞,就是上面的 wait 队列,由被动调度触发)
  ○ 我们直接创建的 g 优先进入当前 p 的本地队列,本地队列满或者主动让渡等情况会进全局队列
● 其他面试问题
  ○ 调度器 P 起到了什么作用?试想没有 P,那么我们 g 直接跟 m 绑定,那么好像其实就跟直接跟使用线程差不多了只是简单封装。首先最明显的作用就是实现了一个 m:n 关系,p 虽然跟处理器没直接关系但是还是确实代表一个抽象,最佳实践就是 p = 核心数(GOMAXPROCS),一定程度上线程的数量。其次这样动态绑定实现了线程的复用。除了这两个设计上的作用,剩下的就是他作为本地队列了,就类似于一个 channel 无锁化的重要步骤。最后还有 p 是作为一个所谓的“万能中间层”,除了动态做这种绑定复用,还能处理僵直的抢占调度的情况,这种情况下 p 会重新绑定一个 m 再去执行 g。
    ■ 有个进阶题目,在一个 GOMAXPROCS 不等于核心数的 container 中运行程序会怎样?简单说,p 的数量直接意味着整个调度系统认为我们 cpu 有多少核,最大并行能力是怎样的。如果多了的话,就是实际线程数可能会比最佳值要大。那么就是调度效率下降,资源竞争加剧,性能下降。少了基本就是没充分利用了。
  ○ 如何实现无锁化的。调度的时候取 g 的顺序,只有全局要加锁,p 是空的时候和窃取失败的时候都会提前取好 g。
  ○ 为什么轻量?用户态,弱内核依赖,上下文切换,栈动态扩缩(初始内存是 0)
  ○ 操作系统如何向某个Goroutine发送时钟中断呢?(这个我暂时没找到答案,目前理解是 os 先通知 go runtime,然后相关 m 在自己执行过程中被插入系统调用。)
  ○ goroutine 终止会对其他 goroutine 产生影响吗(已捕获可恢复可以,否则会因为传递到顶层 g 导致全部崩溃)
  ○ 什么时候被回收(这个问题有点奇怪,了解所有调度方式就好了,正常用完都会被回收,不然就是 panic 恢复,程序结束)
全部评论
好文章
1 回复 分享
发布于 2024-09-05 16:10 重庆
佬太强了
点赞 回复 分享
发布于 2024-09-01 10:19 北京

相关推荐

笔记背景从秋招提前批参与到现在,岗位优先级按表内第二列从左到右依次排序,9月后不能网上查询结果的投递不进行记录,按照工作内容笼统分为4大类:HR、行政专员、销售、项目管理,并且根据大类投递对应简历自身简历1个月校外行政岗实践,2个校内比赛项目,4段学生干部经历(简历中体现两段),奖学金2次国奖2项省奖3项校内荣誉头衔4项数据分析从数据上来看,三本学生特别是工商管理系的同学也能投递人力资源岗(8月份有个同学和我说三本就别投人力了;-D)岗位的难易度是销售>HR>行政专员>项目管理,行政专员虽然没OFFER但是最近有个推进比较迅速的牛马岗从被拒(简历无法过筛)的百分比来看,各位同学如果投递多个简历没有推进的原因不在自身,而在大环境,请多多投递~复筛的占比不大,说明大多数公司并不会因为复筛终止流程,如果大家推进到复筛可以早日做下一步的准备此处笔试是指企业告诉同学在筛选后开展的笔试,从占比上来说笔试是会筛人的(数据上可能是因为我没去银行的笔试所以虚高,不过真被中通还有网龙的笔试卡过,哔哩哔哩也是笔试后被拒)OFFER/二面/一面的百分比大概是1/2/4,请大家好好准备面试,争取通过数据之外大家在投递岗位的时候可以看看大家曝光的薪资待遇,如果没有双非的同学爆料的大厂或者说大企业可能会存在筛选学历的情况网申推进很忙,要多参加线下的宣讲会和招聘会,我有过从8月到现在还在初筛的投递记录同时如果有高三或者以后高考选学校的同学看到的话——想在家或省内工作的同学就不要报省外的学校了,报了就代表线下招聘和你没关系了,没有哪个地区的HR会好心帮其他分公司的HR招的(当然也做不到)写在最后因为之前笔记的内容把投递的公司都写进去了导致有一次居然被官方看到评论了,吓得我拿到OFFER之后完全不敢发出来(指offer对比或者介绍详细一些),所以后续会修改各个笔记的内容作为基恩士和金山办公的校园大使,基恩士的HR态度很好,给了我二面的机会不过我没把握住。金山办公到现在也不知道招不招三本的学生以至于我也不敢和同学说填内推码投递简历,群内的活动和引导做的一坨,把校园大使不当人看除了金山办公我没见过写的乱七八糟的,也谢谢大家能够跟着我投递,因为有offer了之后可能就不怎么更新进度写在最最后真想告诉三本的大家,我们这样的同学要比别人投入秋招的时间早点,只要在路上怎么走都是在前往终点的路上 #牛客创作赏金赛# #你的秋招简历被谁挂了?# #非技术投递记录#
点赞 评论 收藏
分享
不愿透露姓名的神秘牛友
2024-11-06 09:47
点赞 评论 收藏
分享
不愿透露姓名的神秘牛友
2024-11-04 14:43
腾讯云智研发 java 13x16 本科985
点赞 评论 收藏
分享
评论
5
16
分享
牛客网
牛客企业服务