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

深入 Go 底层原理(十):defer 的实现与性能开销

1. 引言

defer 语句是 Go 语言提供的一种用于延迟函数调用的机制,通常用于资源释放、解锁、记录日志等场景。它能保证在函数返回前,无论函数是正常返回还是发生 panicdefer 的调用都会被执行。

defer 使用起来非常方便,但它的实现并非零成本。了解其底层原理有助于我们评估其性能影响。

2. defer 的实现原理

在 Go 的早期版本中,defer 是通过一个链表来实现的。每个 defer 语句都会创建一个 _defer 结构体,并将其插入到当前 goroutine 的 defer 链表的头部。函数返回时,会逆序遍历这个链表并执行调用。这种方式在 defer 数量多时性能较差。

现代 Go 版本 (Go 1.14+) 的优化: 现代 Go 编译器对 defer 进行了大幅优化,大部分 defer 调用现在是基于栈的,而非堆分配。

  • 编译期分析:编译器会分析函数中的 defer 语句。如果 defer 语句在 for 循环之外,且数量固定,编译器会采用基于栈的实现。

  • 栈上分配:编译器会在函数的栈帧上预留空间来记录 defer 的信息(要调用的函数指针和参数)。

  • deferprocdeferreturn:

    • 当执行到 defer 语句时,会调用一个 deferproc 指令,将函数和参数信息存入栈上的预留空间。

    • 在函数返回前,会插入一个 deferreturn 指令,它会检查栈上的 defer 记录,并依次执行它们。

这种方式避免了堆分配和链表操作,性能几乎与直接调用相当。只有在循环中或无法在编译期确定数量的 defer,才会退回到旧的、基于堆分配的实现。

3. defer 的执行时机与陷阱
  • 参数预计算defer 语句的函数参数是在 defer 语句执行时就被计算和固定的,而不是在函数返回时。

    func main() {i := 0defer fmt.Println("Result:", i) // i 的值 0 在这里被固定i++return // 输出: Result: 0
    }
    
  • return 的交互deferreturn 语句之后、函数实际返回之前执行。它可以读取和修改函数的命名返回值

    func getNumber() (result int) {defer func() {result *= 2}()return 5 // 1. result = 5; 2. defer runs: result = 10; 3. return 10
    }
    // getNumber() 返回 10
http://www.xdnf.cn/news/1231669.html

相关文章:

  • hcip---ospf知识点总结及实验配置
  • 淘宝获取商品SKU详情API接口操作指南
  • Python爬虫实战:研究SimpleCV技术,构建图像获取及处理系统
  • 注意点:不同对象(更准确地说,不同类型/类)的魔法方法(Magic Methods,也叫特殊方法,以双下划线`__`开头和结尾)通常是不一样的。
  • 字节Seed发布扩散语言模型,推理速度达2146 tokens/s,比同规模自回归快5.4倍
  • 深入 Go 底层原理(三):Goroutine 的调度策略
  • [论文阅读] 人工智能 + 软件工程 | GitHub Marketplace中CI Actions的功能冗余与演化规律研究
  • Text2SQL:如何通过自然语言直接获取数据,打破技术壁垒?
  • 【Android】通知
  • Docker 的网络模式
  • 红黑树(RBTree)
  • 【LeetCode 热题 100】(四)子串
  • 前端-移动Web-day3
  • 云环境K8s集群WebSocket连接失败解决方案
  • 【REACT18.x】使用vite创建的项目无法启动,报错TypeError: crypto.hash is not a function解决方法
  • 基于 LightGBM 的二手车价格预测
  • GaussDB having 的用法
  • 图像加密学习日志————论文学习DAY4
  • 分布式事务----spring操作多个数据库,事务以及事务回滚还有用吗
  • 机械臂的轨迹生成的多种方案
  • Jupyter notebook如何显示行号?
  • MFC 实现托盘图标菜单图标功能
  • NCV8402ASTT1G自保护N沟道功率MOSFET安森美/ONSEMI 过流过温保护汽车级驱动NCV8402ASTT1
  • 从基础功能到自主决策, Agent 开发进阶路怎么走?
  • 【计算机网络】Socket网络编程
  • Android 15 限制APK包手动安装但不限制自升级的实现方案
  • 断路器瞬时跳闸曲线数据获取方式
  • Javaweb————Apache Tomcat服务器介绍及Windows,Linux,MAC三种系统搭建Apache Tomcat
  • 嵌入式第十八课!!数据结构篇入门及单向链表
  • Oracle 11gR2 Clusterware应知应会