defer 在函数返回前按后进先出顺序执行,参数在 defer 语句执行时快照求值,可修改命名返回值,常用于资源清理与耗时统计。

defer 是 go 语言中用于延迟执行函数调用的关键字,它不改变代码逻辑结构,但能清晰表达“收尾动作”的意图。它的核心作用是:在当前函数即将返回前,按 后进先出(LIFO)顺序 执行所有被 defer 的语句。理解这一点,就抓住了 defer 执行顺序的全部关键。
defer 的执行时机和栈式行为
defer 并不是在函数结束时才注册动作,而是在 执行到 defer 语句那一行时立即注册,但实际调用被推迟到函数 return 前(包括正常 return、panic 中断、或函数自然结束)。多个 defer 会像栈一样压入,所以最后写的 defer 最先执行。
- func f() {
defer fmt.Println(“1”)
defer fmt.Println(“2”)
return
}
输出是:2,然后 1 —— 不是书写顺序,而是注册顺序的逆序。
defer 中变量捕获的是“快照”还是“引用”?
这是最容易出错的地方:defer 表达式中的参数在 defer 语句执行时就被求值并保存(即“快照”),而不是在真正调用时再取值。例如:
立即学习“go语言免费学习笔记(深入)”;
- i := 0
defer fmt.Println(i) // i=0 被捕获
i = 1
return
输出是 0,不是 1。若想捕获运行时值,可改用闭包:
- i := 0
defer func() { fmt.Println(i) }() // i 在调用时读取
i = 1
return
此时输出是 1。
defer 和 return 的配合细节
Go 中 return 不是原子操作:它会先给返回值赋值(如果有命名返回值),再执行 defer,最后跳转离开函数。这意味着 defer 可以修改命名返回值:
- func f() (result int) {
defer func() { result *= 2 }()
return 3
}
该函数返回 6。注意:这个技巧只对命名返回值有效;匿名返回值无法被 defer 修改。
典型实用场景
defer 最自然的用途是资源清理,比如:
- 打开文件后 defer file.Close()
- 加锁后 defer mu.Unlock()
- 启动 goroutine 做清理时 defer cancel()
- 记录耗时:start := time.Now(); defer func(){ log.printf(“took %v”, time.Since(start)) }()
只要动作有明确的“配对生命周期”,且发生在同一函数内,defer 就比手动写在每个 return 前更安全、简洁。
基本上就这些。defer 看似简单,但执行时机、参数求值、与 return 的交互这三点理清了,就不会踩坑。