defer在函数返回前按后进先出顺序执行,参数在defer语句执行时即被求值,循环中直接defer调用循环变量会导致所有调用使用最终值。

Go语言中的defer语句用于延迟函数调用,使其在当前函数返回前执行,常用于资源释放、错误处理等场景。虽然用法简单,但其执行时机和一些边界情况容易引发误解和陷阱。
defer语句注册的函数会在包含它的函数返回之前执行,遵循后进先出(LIFO)的顺序。这意味着多个defer语句会逆序执行。
例如:
func main() {
defer fmt.Println("first")
defer fmt.Println("second")
fmt.Println("normal")
}
normal second first
注意:defer的函数参数在defer语句执行时即被求值,而不是在实际调用时。这是常见陷阱之一。
在for循环中直接defer调用并传入循环变量,可能导致所有defer调用都使用了同一个变量值。
立即学习“go语言免费学习笔记(深入)”;
for i := 0; i < 3; i++ {
defer func() {
fmt.Println(i)
}()
}
输出会是三个3,而不是0、1、2。因为i是引用的外部变量,当defer执行时,循环已结束,i的值为3。
解决方法:通过参数传入或在循环内创建局部变量:
for i := 0; i < 3; i++ {
defer func(i int) {
fmt.Println(i)
}(i)
}
当函数有命名返回值时,defer可以修改返回值,尤其是在使用闭包或指针时。
func f() (result int) {
defer func() {
result++
}()
return 1
}
该函数返回2,因为defer在return赋值后执行,修改了命名返回值result。
但如果return后面有表达式,执行顺序是:先计算返回值,再执行defer,最后函数退出。
defer最典型的用途是确保资源被释放,比如文件关闭、锁释放。
file, err := os.Open("test.txt")
if err != nil {
return err
}
defer file.Close()
即使后续操作发生panic,defer也能保证文件被关闭。但要注意,file.Close()可能返回错误,在生产环境中应处理该错误,例如通过log记录。
另一个常见模式是在函数入口defer recover,用于捕获panic:
defer func() {
if r := recover(); r != nil {
log.Printf("panic recovered: %v", r)
}
}()
基本上就这些。defer强大但需小心使用,理解其执行时机和变量绑定机制能避免多数问题。
以上就是Golang的defer语句执行时机和常见应用陷阱的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号