当前位置: 首页 > 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.lryc.cn/news/429690.html

相关文章:

  • C库函数signal()信号处理
  • 007 SpringCloudAlibaba基础使用(nacos,gateway)
  • 编译环境揭秘
  • 不同的字符集(ASCII、UTF-8、UTF-16/UCS-2、UTF-32/UCS-4)
  • STM32f407 网络接收 fpga 的 bin 文件并更新到 fpga series7(3)
  • JavaScript基础知识(七)
  • 20240821让飞凌的OK3588-C的核心板在Linux R4下挂载1TB的exFAT格式的TF卡
  • Java HashMap练习
  • 前后端分离项目实战-通用管理系统搭建(前端Vue3+ElementPlus,后端Springboot+Mysql+Redis)第三篇:登录功能优化
  • 8.20 Redis ACL配置 多个用户连接同一个Redis
  • 【C语言】static和extern的作用
  • 全新分支版本!微软推出Windows 11 Canary Build 27686版
  • 【Linux】ARM服务器命令行安装虚拟机
  • Android 10.0 锁屏页面忘记锁屏密码情况下点击5次解锁图标弹出锁屏密码功能实现
  • Java-CompletableFuture工具类
  • C语言:递归
  • 自动化测试框架pytest+allure+requests
  • Python 笔记 numpy.ndarray切片
  • 一、HTML5知识点精讲
  • 【杂乱算法】前缀和与差分
  • Arduino调试ESP32常见问题 exit status 1
  • “决胜面试:高频题目与算法策略一览”
  • Node-RED的安装
  • java中的Collections
  • linux Qt QkeyEvent及驱动键盘按键捕获
  • 【GH】【EXCEL】P6: Shapes
  • google浏览器chrome用户数据(拓展程序,书签等)丢失问题
  • 数据结构——链式队列和循环队列
  • 数据库死锁解决方法,学费了吗?
  • API网关之Apache ShenYu