深入理解Go中defer的机制
defer 是 Go 语言中用于延迟执行函数调用的关键字,常用于资源清理(如关闭文件、释放锁)和异常处理。但其行为机制存在一些隐蔽的细节,稍有不慎可能导致难以察觉的 Bug。本文通过多个直观示例,深入剖析 defer 的核心机制。
一、defer 的执行顺序:后进先出(LIFO)
多个 defer 语句按逆序执行,类似于栈的“后进先出”原则。
示例 :多个 defer 的执行顺序
func main() {
defer fmt.Println("defer 1")
defer fmt.Println("defer 2")
fmt.Println("main 逻辑")
}
输出:
main 逻辑
defer 2
defer 1
结论:
defer语句按注册顺序的逆序执行,确保依赖资源按正确顺序释放(如先打开的文件后关闭)。
二、defer 的参数预计算:值拷贝的陷阱
defer 的参数在注册时即被预计算并拷贝,而非执行时动态获取。
示例 :参数预计算的影响
func main() {
x := 10
defer fmt.Println("defer 中的 x:", x) // x 的值在注册时被拷贝
x = 20
fmt.Println("main 中的 x:", x)
}
输出:
main 中的 x: 20
defer 中的 x: 10
结论:
- 若参数是值类型(如
int、string),defer会拷贝当前值,后续修改不影响已注册的defer。 - 若参数是指针或引用类型(如
*int、slice),拷贝的是地址,后续修改会影响defer的执行结果。
三、defer 与闭包:动态绑定的变量
defer 函数若使用外部变量(闭包),会引用变量的最新值,而非注册时的值。
示例 :闭包中的变量绑定
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:值返回与指针返回的差异
// 值返回: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通过地址修改原变量,影响最终结果。
到此这篇关于Go中defer的机制的文章就介绍到这了,更多相关Go defer机制内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
解决golang处理http response碰到的问题和需要注意的点
这篇文章主要介绍了解决golang处理http response碰到的问题和需要注意的点,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧2020-12-12
解决Go Json Unmarshal反序列化丢失数字精度问题
业务会使用 id生成器 产生的 分布式唯一ID,长度比较长,所以代码反序列化时,会出现精度丢失问题,那如何解决呢,下面小编就来和大家详细讲讲2023-08-08
Golang时间处理库go-carbon v2.2.13发布细则
这篇文章主要为大家介绍了Golang 时间处理库go-carbon v2.2.13发布细则,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪2023-11-11


最新评论