defer用于延迟执行函数调用,确保资源释放;多个defer按后进先出顺序执行;结合recover可处理panic,常用于文件、锁、网络连接等场景的清理工作。

在Go语言中,defer 是一个非常实用的关键字,用于延迟执行某个函数调用,直到包含它的函数即将返回时才执行。这个特性非常适合用来保证资源的释放,比如关闭文件、释放锁、关闭网络连接等,避免因遗漏而导致资源泄漏。
defer 的基本用法
使用 defer 非常简单:只要在函数调用前加上 defer 关键字,该函数就会被推迟到当前函数 return 之前执行。
例如:打开一个文件并确保它最终被关闭:
func readFile(filename string) error {
file, err := os.Open(filename)
if err != nil {
return err
}
defer file.Close() // 函数返回前自动调用
// 使用 file 进行读取操作
scanner := bufio.NewScanner(file)
for scanner.Scan() {
fmt.Println(scanner.Text())
}
return scanner.Err()
}
即使函数中途发生错误或提前 return,file.Close() 依然会被执行,从而确保文件描述符不会泄漏。
立即学习“go语言免费学习笔记(深入)”;
多个 defer 的执行顺序
如果在一个函数中使用了多个 defer,它们会按照“后进先出”(LIFO)的顺序执行。
示例:func example() {
defer fmt.Println("first")
defer fmt.Println("second")
defer fmt.Println("third")
}
// 输出:
// third
// second
// first
这种特性在需要按顺序释放资源时特别有用,比如逐层解锁或反向清理。
结合 panic 和 recover 使用
defer 不仅在正常流程中有效,在发生 panic 时也能保证执行,因此适合做兜底清理工作。
比如在网络服务中释放连接或记录日志:
func handleConnection(conn net.Conn) {
defer func() {
if r := recover(); r != nil {
log.Printf("panic caught: %v", r)
}
conn.Close() // 即使出现 panic,连接也会被关闭
log.Println("connection closed")
}()
// 处理请求逻辑,可能触发 panic
process(conn)
}
常见使用场景
- 文件操作:打开后立即 defer file.Close()
- 互斥锁:加锁后 defer mu.Unlock()
- 数据库连接/事务:开始事务后 defer tx.Rollback() 或 defer db.Close()
- HTTP 响应体:resp.Body 使用后 defer resp.Body.Close()
注意:对于 resp.Body 等 io.ReadCloser,即使读取失败也应关闭,defer 能有效覆盖所有退出路径。
基本上就这些。合理使用 defer 能显著提升代码的安全性和可读性,让资源管理更简洁可靠。关键是:在获取资源后立刻写 defer 释放语句,不要拖延。










