在 go 中,处理错误的模式有:error 值类型:使用 error interface 值表示错误。errors 包:提供 new、wrap、unwrap 函数对错误进行创建、包装和拆包。sentinel 错误:使用特殊错误值表示特定错误类型,避免字符串比较。
Go 中错误处理的不同设计模式
在 Go 中,错误处理是一个至关重要的方面,它可以帮助您编写健壮、易于维护的代码。本文将探讨在 Go 中处理错误的不同设计模式,并通过实际示例来展示它们的用法。
模式一:error 值类型
这是 Go 中处理错误的最基本的方式。error 值是一个特殊的 interface 类型,实现该接口的值表示一个错误。
func f() error {
// ...
return errors.New("some error")
}
func main() {
if err := f(); err != nil {
fmt.Println(err)
}
}
模式二:errors 包
errors 包提供了几个有用的函数和类型来帮助处理错误,包括:
- New: 创建一个新的 error 值
- Wrap: 将一个错误包装在另一个错误中
- Unwrap: 从一个错误中提取包装错误
func f() error {
// ...
return errors.Wrap(errors.New("some error"), "wrapper error")
}
func main() {
if err := f(); err != nil {
fmt.Println(errors.Unwrap(err)) // "some error"
}
}
模式三:sentinel 错误
sentinel 错误是一种特殊的错误值,用于表示特定类型的错误。通过使用 sentinel 错误,您可以避免使用字符串比较或其他昂贵的检查。
type MyError struct{}
func (e *MyError) Error() string { return "my error" }
func f() error {
// ...
if something {
return MyError{}
}
// ...
}
func main() {
if err := f(); err != nil {
if _, ok := err.(MyError); ok {
// Handle "my error" here
}
}
}
实战案例
以下是一段使用 sentinel 错误模式的代码:
func createAccount(user *User) (*Account, error) {
// ...
if user.Age < 18 {
return nil, &MinorError{}
}
// ...
}
type MinorError struct{}
func (e *MinorError) Error() string { return "cannot create account for minors" }
func main() {
user := &User{Age: 17}
_, err := createAccount(user)
if err != nil {
if _, ok := err.(*MinorError); ok {
fmt.Println("User is too young to create an account.")
} else {
fmt.Println(err)
}
}
}