深入 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 语句执行时就被计算和固定的,而不是在函数返回时。

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

    Go 复制代码
    func getNumber() (result int) {
        defer func() {
            result *= 2
        }()
        return 5 // 1. result = 5; 2. defer runs: result = 10; 3. return 10
    }
    // getNumber() 返回 10
相关推荐
澜莲花13 小时前
python图色之opencv基础---验证码实战
开发语言·python·opencv
沐知全栈开发14 小时前
Numpy 数组操作
开发语言
yaoxin52112314 小时前
279. Java Stream API - Stream 拼接的两种方式:concat() vs flatMap()
java·开发语言
@小码农14 小时前
202512 电子学会 Scratch图形化编程等级考试三级真题(附答案)
服务器·开发语言·数据结构·数据库·算法
Cosmoshhhyyy14 小时前
《Effective Java》解读第29条:优先考虑泛型
java·开发语言
一路往蓝-Anbo14 小时前
C语言从句柄到对象 (六) —— 继承与 HAL:父类指针访问子类数据
c语言·开发语言·stm32·嵌入式硬件·物联网
北冥有一鲲14 小时前
A2A协议与LangChain.js实战:构建微型软件工厂
开发语言·javascript·langchain
Chen不旧14 小时前
java基于reentrantlock/condition/queue实现阻塞队列
java·开发语言·signal·reentrantlock·await·condition
laplace012315 小时前
Part 3:模型调用、记忆管理与工具调用流程(LangChain 1.0)笔记(Markdown)
开发语言·人工智能·笔记·python·langchain·prompt
风送雨15 小时前
八周Python强化计划(七)
开发语言·python