defer关键字用于延迟执行函数调用,确保在函数返回前执行资源清理等操作;多个defer按后进先出顺序执行。

在Golang中,defer 是一个非常实用的关键字,用于延迟执行某个函数调用,直到包含它的函数即将返回时才执行。它常用于资源清理、解锁、关闭文件等场景,确保关键操作不会被遗漏。
defer的基本语法和执行顺序
使用 defer 非常简单,只需在函数调用前加上 defer 关键字:
defer fmt.Println("world")
fmt.Println("hello")
输出结果为:
hello world
可以看到,defer 的语句虽然写在前面,但执行被推迟到了当前函数返回前。如果有多个 defer,它们会按照“后进先出”(LIFO)的顺序执行:
立即学习“go语言免费学习笔记(深入)”;
for i := 0; i < 3; i++ {
defer fmt.Println(i)
}
输出为:
2 1 0
defer在资源管理中的典型应用
最常见的是在打开文件后立即使用 defer 来关闭文件:
file, err := os.Open("data.txt")
if err != nil {
log.Fatal(err)
}
defer file.Close()
// 处理文件内容
data := make([]byte, 100)
file.Read(data)
fmt.Println(string(data))
即使后续代码发生 panic 或提前 return,defer 都能保证 file.Close() 被调用,避免资源泄漏。
类似的场景还包括:
- 释放互斥锁:
defer mutex.Unlock() - 关闭数据库连接:
defer db.Close() - 关闭网络连接:
defer conn.Close()
defer与函数参数求值时机
需要注意的是,defer 后面的函数参数在 defer 执行时就立即求值,但函数调用本身延迟执行:
func main() {
i := 1
defer fmt.Println(i) // 输出 1,不是 2
i++
return
}
上面例子中,尽管 i 后续被修改为 2,但 defer 捕获的是当时 i 的值(1)。
如果想延迟求值,可以使用匿名函数:
i := 1
defer func() {
fmt.Println(i) // 输出 2
}()
i++
defer的注意事项
虽然 defer 很方便,但也有一些细节需要注意:
- defer 语句必须出现在函数体中,不能在全局作用域使用
- 在循环中大量使用 defer 可能影响性能,因为每个 defer 都会加入栈中
- defer 无法跳过 panic,但它会在 panic 触发后、函数返回前执行,可用于记录日志或恢复
- 配合 recover 使用时,defer 必须定义在 panic 发生之前
基本上就这些。合理使用 defer 能让代码更安全、简洁,尤其在处理资源管理和错误控制时特别有用。关键是理解它的执行时机和参数捕获行为,避免误用。










