defer语句在函数返回前按后进先出顺序执行,参数在声明时即求值,常用于资源释放和清理操作。

在Go语言中,defer语句用于延迟函数的执行,直到包含它的函数即将返回时才执行。它常用于资源释放、清理操作等场景,比如关闭文件、解锁互斥锁等。理解defer的语法和执行顺序对编写正确的Go代码非常重要。
defer基本语法
defer后跟一个函数或方法调用:
defer functionName()
也可以带参数:
defer fmt.Println("done")
注意:defer后面的函数参数在defer语句执行时就会被求值,但函数本身要等到外层函数返回前才被调用。
立即学习“go语言免费学习笔记(深入)”;
多个defer的执行顺序
当一个函数中有多个defer语句时,它们的执行顺序是后进先出(LIFO),即最后声明的defer最先执行。
示例:
func main() {
defer fmt.Println("first")
defer fmt.Println("second")
defer fmt.Println("third")
}
输出结果为:
third second first
这是因为三个defer被压入栈中,函数返回时依次弹出执行。
defer参数求值时机
defer语句中的函数参数在defer被执行时立即求值,而不是在函数实际调用时。
示例:
func example() {
i := 1
defer fmt.Println(i) // 输出 1,不是2
i++
return
}
尽管i在defer之后被递增,但fmt.Println(i)中的i在defer语句执行时已经确定为1。
如果想延迟求值,可以使用匿名函数:
func example() {
i := 1
defer func() {
fmt.Println(i) // 输出 2
}()
i++
return
}
常见使用场景
defer最典型的用途包括:
- 关闭文件:defer file.Close()
- 释放锁:defer mu.Unlock()
- 记录函数耗时:
func slowOperation() {
defer logDuration(time.Now())
// 模拟耗时操作
}
func logDuration(start time.Time) {
log.Printf("operation took %v", time.Since(start))
}
基本上就这些。掌握defer的关键在于记住两点:参数求值时机和执行顺序为后进先出。合理使用defer可以让代码更简洁、安全。不复杂但容易忽略细节。










