defer关键字用于延迟执行函数调用,确保资源如文件、锁、网络连接等在函数返回前被释放,遵循后进先出原则,多个defer按声明逆序执行,可结合命名返回值捕获panic并修改错误,常用于文件操作、网络连接、锁释放和数据库事务中,提升代码健壮性和可读性。

在go语言中,defer 是一个非常实用的关键字,常用于资源释放和错误处理。它能确保某些操作(如关闭文件、释放锁、关闭网络连接等)在函数返回前执行,无论函数是正常结束还是因错误提前退出。合理使用 defer 可以提升代码的健壮性和可读性。
defer 的基本用法
defer 语句会将后面的函数调用延迟到当前函数返回之前执行。它的执行遵循“后进先出”原则。
例如,打开文件后需要及时关闭:
<pre class="brush:php;toolbar:false;">func readFile(filename string) error { file, err := os.Open(filename) if err != nil { return err } defer file.Close() // 函数结束前自动关闭 // 读取文件内容 scanner := bufio.NewScanner(file) for scanner.Scan() { fmt.Println(scanner.Text()) } return scanner.Err() }
这里 file.Close() 被 defer 延迟执行,即使后续出现错误或提前 return,文件也能被正确关闭。
多个 defer 的执行顺序
当一个函数中有多个 defer 时,它们按声明的逆序执行。
立即学习“go语言免费学习笔记(深入)”;
<pre class="brush:php;toolbar:false;">func example() { defer fmt.Println("first") defer fmt.Println("second") defer fmt.Println("third") } // 输出:third → second → first
这种特性适合处理嵌套资源释放,比如多层锁或多个文件操作。
defer 与错误处理的结合
defer 不仅用于资源管理,还可以配合命名返回值捕获和修改错误。
例如,在函数中记录错误日志或进行恢复:
<pre class="brush:php;toolbar:false;">func riskyOperation() (err error) { mutex.Lock() defer mutex.Unlock() defer func() { if r := recover(); r != nil { err = fmt.Errorf("panic recovered: %v", r) } }() // 模拟可能 panic 的操作 result := 10 / 0 // 实际运行会 panic _ = result return nil }
在这个例子中,defer 配合匿名函数实现了对 panic 的捕获,并通过命名返回值修改了最终返回的错误。
常见使用场景
- 文件操作:打开后立即 defer Close()
- 网络连接:建立连接后 defer conn.Close()
- 锁的释放:加锁后 defer mu.Unlock()
- 数据库事务:开始事务后根据 err 决定 Commit 或 Rollback,也可结合 defer 简化逻辑
基本上就这些。defer 让资源管理和错误兜底变得更简单,只要记得在获取资源后尽早写上 defer 释放语句,就能避免大多数泄漏问题。


