defer 不直接处理错误,而是确保资源清理在函数返回前执行;它按LIFO顺序延迟调用清理函数,参数在声明时求值,适用于解耦错误检查与资源释放。
在 Go 中,defer 本身不直接“处理错误”,但它能确保资源清理逻辑在函数返回前执行,无论是否发生错误。关键在于把错误检查和资源释放解耦:先用 defer 注册清理动作(如关闭文件、释放锁),再在合适位置显式检查并返回错误。
defer 语句会在其所在函数即将返回(包括正常 return、panic 或提前 return)时按后进先出(LIFO)顺序执行。它不关心函数是否出错,只保证“该做的事做完”。比如打开一个文件后立即 defer 关闭,就无需在每个 error 分支里重复写 f.Close()。
以读取文件为例:
func readFile(filename string) ([]byte, error) {
f, err := os.Open(filename)
if err != nil {
return nil, err // 错误直接返回
}
defer f.Close() // 确保函数退出前关闭
data, err := io.ReadAll(f)
if err != nil {
return nil, fmt.Errorf("read %s: %w", filename, err)
}
return data, nil // 此处返回,f.Close() 自动触发
}
注意:defer f.Close() 写在 open 成功之后,避免对 nil 文件句柄调用 Close;同时不检查 f.Close() 的返回值——因为此时函数已准备返回,Close 失败通常只能记录日志,不能改变主逻辑的错误结果。
有些资源释放本身可能失败(如数据库事务回滚、网络连接强制关闭),且该失败对业务有意义。这时可将清理封装成函数,并在 defer 中调用,再单独处理其错误:
func processDB() error {
tx, err := db.Begin()
if err != nil {
return err
}
// 封装清理逻辑,支持返回错误
cleanup := func() {
if rErr := tx.Rollback(); rErr != nil {
log.Printf("rollback failed: %v", rErr)
// 这里不覆盖原始 err,除非你明确想优先报告清理失败
}
}
defer cleanup()
// 执行业务操作...
if _, err := tx.Exec("INSERT ..."); err != nil {
return err // rollback 会在 defer 中自动触发
}
return tx.Commit() // 成功则提交,cleanup 中的 Rollback 不生效(需改写逻辑)
}
更稳妥的做法是:用命名返回值 + defer 组合,在 defer 中根据函数最终返回的 error 决定执行 Commit 还是 Rollback:
func processDB() (err error) {
tx, err := db.Begin()
if err != nil {
return
}
defer func() {
if err != nil {
if rErr := tx.Rollback(); rErr != nil {
log.Printf("rollback failed: %v", rErr)
}
} else {
err = tx.Commit()
}
}()
_, err = tx.Exec("INSERT ...")
return // err 由 defer 函数统一处理
}
defer resp.Body.Close() 是标准写法,但 defer json.NewEncoder(w).Encode(data) 可能掩盖编码失败
于需要即时响应的错误恢复:它无法替代 if-else 错误分支或 recover —— panic 后的 defer 会执行,但程序已脱离正常流程