当前位置: 首页 > news >正文

Go 语言协程管理精解

1.基础

        协程切换需要操作寄存器,这些操作需要通过汇编辅助实现。另外,每一个协程都有一个协程栈,实际上协程栈也是有结构的。汇编程序和栈结构这些概念可能大部分开发者都不太了解,在介绍协程管理之间,先简要介绍。

1.1 汇编入门

学习 Go 语言协程还需要掌握汇编程序吗?其实不需要对汇编多么熟悉,只需要简单了解常用的一些汇编指令即可。 

Go 语言本身就提供了很多工具,例如,编译工具 compile 用于编译 Go 程序,我们可以使用它将上述 Go 程序编译为汇编代码。编译命令以及编译后的汇编代码如下:

//-N 禁止优化 -l 禁止内联 -S 输出汇编
go tool compile -S -N -l test.go
// addSub 函数编译后的汇编代码
"".addSub STEXT nosplit size=49 args=0x20 locals=0x00x0000 00000 (test.go:3)    MOVQ    $0, "".~r2+24(SP)0x0009 00009 (test.go:3)    MOVQ    $0, "".~r3+32(SP)0x0012 00018 (test.go:4)    MOVQ    "".a+8(SP), AX0x0017 00023 (test.go:4)    ADDQ    "".b+16(SP), AX0x001c 00028 (test.go:4)    MOVQ    AX, "".~r2+24(SP)0x0021 00033 (test.go:4)    MOVQ    "".a+8(SP), AX0x0026 00038 (test.go:4)    SUBQ    "".b+16(SP), AX0x002b 00043 (test.go:4)    MOVQ    AX, "".~r3+32(SP)0x0030 00048 (test.go:4)    RET// main 函数编译后的汇编代码
"".main STEXT size=68 args=0x0 locals=0x280x000f 00015 (test.go:7)      SUBQ    $40, SP0x0013 00019 (test.go:7)      MOVQ    BP, 32(SP)0x0018 00024 (test.go:7)      LEAQ    32(SP), BP0x001d 00029 (test.go:8)      MOVQ    $333, (SP)0x0025 00037 (test.go:8)      MOVQ    $222, 8(SP)0x002e 00046 (test.go:8)      CALL    "".addSub(SB)0x0033 00051 (test.go:9)      MOVQ    32(SP), BP0x0038 00056 (test.go:9)      ADDQ    $40, SP0x003c 00060 (test.go:9)      RET

如下所示:

协程退出

        协程的入口函数为 gofunc,执行完成时,最后一条语句是 “RET” 汇编指令,它将从协程栈弹出 8 字节数据,并存储到程序计数器 PC,随后通过 "JMP" 指令跳转。“RET” 弹出的是函数 runtime.goexit 首地址,就相当于跳转到了函数 runtime.goexit,该函数代码如下:

//函数 runtime.goexit 是汇编代码实现的,调用了函数 runtime.goexit1
void goexit1(void){mcall(goexit0)
}//系统栈执行该函数
func goexit0(gp *g){//设置协程状态,执行回收操作casgstatus(gp,_Grunning,_Gdead)//省略了清理协程相关数据的逻辑//添加到空闲队列gfput(_p_,gp)//调度schedule()
}

        需要注意的是,函数 runtime.goexit 是汇编代码实现的,底层直接调用了函数 runtime.goexit1。同样,这里是通过函数 runtime.mcall 切换到系统栈,所以函数 runtime.goexit0 是在系统栈执行的,也是它完成的协程的收尾工作,包括修改协程状态为_Gdead,清理协程相关数据,将协程回收到逻辑处理器 P 的空闲队列,执行调度程序等。


http://www.mrgr.cn/news/4141.html

相关文章:

  • 基于缓存提高Java模板文件处理性能:减少磁盘I/O的实践与探索
  • DMDSC搭建
  • 软件测试学习笔记丨多表查询及子查询
  • React 学习——React.memo(简单、引用类型的prop)
  • Maven的简单使用
  • maven进阶(超详细)
  • [论文笔记]ZeRO: Memory Optimizations Toward Training Trillion Parameter Models
  • etcd参数解释
  • 《艺术与设计》是什么级别的期刊?是正规期刊吗?能评职称吗?
  • 18款各具特色的项目管理系统软件,项目经理用过都说好!
  • Unity XR Interaction Toolkit 通过两个手柄控制物体放大缩小
  • 开源低代码开发工具Lowcoder
  • VTK—vtkCutter截取平面数据
  • 嵌入式AI快速入门课程-K510篇 (第二篇 Ubuntu的基础操作)
  • uniapp 获取定位权限及问题解决
  • wordrpess心得
  • 【图文并茂】ant design pro 如何给后端发送 json web token - 请求拦截器的使用
  • 设计模式 -- 七大原则(二)-- 接口隔离原则
  • LeetCode216 组合总和 III
  • 十三、OpenCVSharp的目标检测