Go 中 defer 的机制

文章目录

    • [Go 语言中 `defer` 的底层机制与实战解析](#Go 语言中 defer 的底层机制与实战解析)
      • [一、`defer` 的执行顺序:后进先出(LIFO)](#一、defer 的执行顺序:后进先出(LIFO))
        • [示例 :多个 `defer` 的执行顺序](#示例 :多个 defer 的执行顺序)
      • [二、`defer` 的参数预计算:值拷贝的陷阱](#二、defer 的参数预计算:值拷贝的陷阱)
        • [示例 :参数预计算的影响](#示例 :参数预计算的影响)
      • [三、`defer` 与闭包:动态绑定的变量](#三、defer 与闭包:动态绑定的变量)
        • [示例 :闭包中的变量绑定](#示例 :闭包中的变量绑定)
      • [四、`defer` 与返回值:隐式的赋值逻辑](#四、defer 与返回值:隐式的赋值逻辑)
        • [示例 4:值返回与指针返回的差异](#示例 4:值返回与指针返回的差异)

Go 语言中 defer 的底层机制与实战解析

defer 是 Go 语言中用于延迟执行函数调用的关键字,常用于资源清理(如关闭文件、释放锁)和异常处理。但其行为机制存在一些隐蔽的细节,稍有不慎可能导致难以察觉的 Bug。本文通过多个直观示例,深入剖析 defer 的核心机制。


一、defer 的执行顺序:后进先出(LIFO)

多个 defer 语句按逆序执行,类似于栈的"后进先出"原则。

示例 :多个 defer 的执行顺序
go 复制代码
func main() {
    defer fmt.Println("defer 1")
    defer fmt.Println("defer 2")
    fmt.Println("main 逻辑")
}

输出:

复制代码
main 逻辑
defer 2
defer 1

结论

  • defer 语句按注册顺序的逆序执行,确保依赖资源按正确顺序释放(如先打开的文件后关闭)。

二、defer 的参数预计算:值拷贝的陷阱

defer 的参数在注册时即被预计算并拷贝,而非执行时动态获取。

示例 :参数预计算的影响
go 复制代码
func main() {
    x := 10
    defer fmt.Println("defer 中的 x:", x) // x 的值在注册时被拷贝
    x = 20
    fmt.Println("main 中的 x:", x)
}

输出:

复制代码
main 中的 x: 20
defer 中的 x: 10

结论

  • 若参数是值类型(如 intstring),defer 会拷贝当前值,后续修改不影响已注册的 defer
  • 若参数是指针或引用类型(如 *intslice),拷贝的是地址,后续修改会影响 defer 的执行结果。

三、defer 与闭包:动态绑定的变量

defer 函数若使用外部变量(闭包),会引用变量的最新值,而非注册时的值。

示例 :闭包中的变量绑定
go 复制代码
func main() {
    x := 10
    defer func() {
        fmt.Println("defer 中的 x:", x) // 闭包引用最新值
    }()
    x = 20
    fmt.Println("main 中的 x:", x)
}

输出:

复制代码
main 中的 x: 20
defer 中的 x: 20

结论

  • 闭包中的变量在 defer 执行时才求值,因此会反映变量的最终状态。
  • 若需固定闭包中的值,需在注册时通过参数传递(如 defer func(a int) { ... }(x))。

四、defer 与返回值:隐式的赋值逻辑

defer 中修改返回值的行为取决于返回值的定义方式(值返回 vs 指针返回)。

示例 4:值返回与指针返回的差异
go 复制代码
// 值返回:defer 修改不影响返回值
func f1() int {
    x := 10
    defer func() { x++ }()
    return x // 实际返回的是 x 的拷贝
}

// 指针返回:defer 修改影响返回值
func f2() *int {
    x := 10
    defer func() { x++ }()
    return &x // 返回 x 的地址
}

func main() {
    fmt.Println(f1()) // 输出 10
    fmt.Println(*f2()) // 输出 11
}

结论

  • 值返回 :返回值在 return 时被拷贝,defer 修改原变量不影响已拷贝的值。
  • 指针返回 :返回的是变量地址,defer 通过地址修改原变量,影响最终结果。

若有错误与不足请指出,关注DPT一起进步吧!!!

相关推荐
C++ 老炮儿的技术栈5 小时前
UDP 与 TCP 的区别是什么?
开发语言·c++·windows·算法·visual studio
wgslucky5 小时前
Dubbo报错:module java.base does not “opens java.lang“ to unnamed module
java·开发语言·dubbo
whyeekkk5 小时前
python打卡第48天
开发语言·python
DougLiang6 小时前
关于easyexcel动态下拉选问题处理
java·开发语言
烛阴7 小时前
bignumber.js深度解析:驾驭任意精度计算的终极武器
前端·javascript·后端
服务端技术栈7 小时前
电商营销系统中的幂等性设计:从抽奖积分发放谈起
后端
全职计算机毕业设计7 小时前
基于Java Web的校园失物招领平台设计与实现
java·开发语言·前端
你的人类朋友7 小时前
✍️Node.js CMS框架概述:Directus与Strapi详解
javascript·后端·node.js
5:007 小时前
云备份项目
linux·开发语言·c++
面朝大海,春不暖,花不开8 小时前
自定义Spring Boot Starter的全面指南
java·spring boot·后端