答案:fmt.Errorf用于创建格式化错误,支持变量插入和错误包装。示例中divide函数用fmt.Errorf返回除零错误,输出“无法除以零:a=10, b=0”;go 1.13+支持%w包装错误,readFile和processFile形成错误链,errors.Is可判断原始错误“权限不足”;自定义错误类型ValidationError结合fmt.Errorf可构造结构化错误信息。
在 Go 语言中,fmt.Errorf 是创建带有格式化信息的错误的常用方式。它允许你像使用 fmt.printf 一样构造错误消息,同时返回一个 error 类型。
基本用法:使用 fmt.Errorf 构造简单错误
最常见的情况是将变量值嵌入错误消息中,便于调试和日志记录。
package main import ( "fmt" ) func divide(a, b int) (int, error) { if b == 0 { return 0, fmt.Errorf("无法除以零:a=%d, b=%d", a, b) } return a / b, nil } func main() { _, err := divide(10, 0) if err != nil { fmt.Println("错误:", err) } }
输出:
错误: 无法除以零:a=10, b=0
嵌套错误(Go 1.13+):使用 %w 包装错误
从 Go 1.13 开始,fmt.Errorf 支持使用 %w 动词来包装原始错误,这样可以保留错误链,便于后续用 errors.Is 和 errors.As 判断。
package main import ( "errors" "fmt" ) func readFile(filename string) error { if filename == "" { return fmt.Errorf("文件名不能为空") } return fmt.Errorf("读取文件失败: %w", errors.New("权限不足")) } func processFile() error { return fmt.Errorf("处理文件时出错: %w", readFile("")) } func main() { err := processFile() fmt.Println("错误:", err) // 判断是否包含特定错误 if errors.Is(err, errors.New("权限不足")) { fmt.Println("原始错误是 '权限不足'") } }
输出:
错误: 处理文件时出错: 读取文件失败: 权限不足
原始错误是 ‘权限不足’
结合自定义错误类型使用
你也可以在自定义错误结构中使用 fmt.Errorf 来生成更丰富的错误信息。
立即学习“go语言免费学习笔记(深入)”;
package main import ( "fmt" ) type ValidationError struct { Field string Msg string } func (e *ValidationError) Error() string { return fmt.Sprintf("验证错误:字段 %s, 原因: %s", e.Field, e.Msg) } func validateAge(age int) error { if age < 0 { return fmt.Errorf("年龄无效: %d, %w", age, &ValidationError{"Age", "不能为负数"}) } return nil } func main() { err := validateAge(-5) if err != nil { fmt.Println(err) } }
输出:
年龄无效: -5, 验证错误:字段 Age, 原因: 不能为负数
基本上就这些。fmt.Errorf 简单但强大,合理使用 %w 可以构建清晰的错误链,对调试和错误处理非常有帮助。
评论(已关闭)
评论已关闭