defer关键字用于延迟执行函数,确保资源释放等操作在函数返回前执行;多个defer按后进先出顺序执行;参数在defer声明时求值,闭包可实现延迟取值;常配合recover捕获panic。

在Go语言中,defer 是一个非常有用的关键字,用于延迟函数或方法的执行。它常被用来做资源释放、锁的解锁、文件关闭等操作,确保这些动作在函数结束前一定被执行,无论函数是正常返回还是发生了panic。
defer的基本用法
使用 defer 非常简单:只要在调用函数或方法前加上 defer 关键字,该调用就会被推迟到包含它的函数即将返回时才执行。
例如:
func example() {file, err := os.Open("test.txt")
if err != nil {
log.Fatal(err)
}
defer file.Close() // 延迟关闭文件
// 其他操作...
}
上面代码中,file.Close() 会在 example 函数返回前自动执行,无需手动在每个退出路径上调用。
立即学习“go语言免费学习笔记(深入)”;
多个defer的执行顺序
当一个函数中有多个 defer 语句时,它们的执行顺序是后进先出(LIFO),也就是最先定义的 defer 最晚执行。
看一个例子:
func multipleDefer() {defer fmt.Println("first defer")
defer fmt.Println("second defer")
defer fmt.Println("third defer")
fmt.Println("function body")
}
输出结果为:
function bodythird defer
second defer
first defer
这说明 defer 是像栈一样被压入和弹出的:每遇到一个 defer,就将其压入栈;函数返回前,从栈顶开始依次执行。
defer与函数参数求值时机
需要注意的是,defer 后面的函数虽然执行被延迟了,但其参数是在 defer 被声明时就立即求值的。
示例:
i := 10
defer fmt.Println(i) // 此时 i 的值是 10,已确定
i = 20
fmt.Println("in function:", i)
}
输出:
in function: 2010
尽管 i 最终变成了 20,但 defer 打印的是 10,因为 fmt.Println(i) 中的 i 是在 defer 语句执行时取值的。
如果想延迟取值,可以使用匿名函数:
func deferWithFunc() {i := 10
defer func() {
fmt.Println(i)
}()
i = 20
fmt.Println("in function:", i)
}
输出:
in function: 2020
此时 defer 执行的是一个闭包函数,i 的值在函数真正执行时才读取,因此输出的是 20。
defer在panic恢复中的作用
defer 经常配合 recover 使用,用于捕获 panic 并防止程序崩溃。
例如:
func safeDivide(a, b int) {defer func() {
if r := recover(); r != nil {
fmt.Println("panic recovered:", r)
}
}()
if b == 0 {
panic("division by zero")
}
fmt.Println("result:", a/b)
}
即使发生 panic,defer 中的 recover 也能捕获并处理,保证函数优雅退出。
基本上就这些。defer 不复杂但容易忽略细节,掌握好它的执行时机和顺序,能写出更安全、清晰的Go代码。










