- 首先我们自定义了ForbiddenError的错误类型
- 我们实现了error接口
- 访问数据库抛出原始错误
- 上层返回ForbiddenError类型的错误
go run 9.goForbidden: permission denied
当然我们也可以不用创建自定义错误的类型,去包装错误添加上下文:package mainimport ( "fmt" "github.com/pkg/errors")type Courseware struct { Id int64 Code string Name string}func getCourseware(id int64) (*Courseware, error) { courseware, err := getFromDB(id) if err != nil {return nil, fmt.Errorf("another wrap err: %w", err) // 1 } return courseware, nil}func getFromDB(id int64) (*Courseware, error) { return nil, errors.New("permission denied")}func main() { _, err := getCourseware(11) if err != nil {fmt.Println(err) }}
- 使用%w包装错误
还有一种方式是使用:
return nil, fmt.Errorf("another wrap err: %v", err)
%v的方式不会包装错误,所以无法追溯到源错误,但往往有时候我们会选择这种方式,而不用%w的方式 。%w的方式虽然能包装源错误,但往往我们会通过源错误去做一些处理,假如源错误被修改,那包装这个源错误的相关错误都需要做响应变化 。3、错误类型判断我们扩展一下上面查询课件的例子 。现在我们有这样的判断,如果传进来的id不合法我们返回400错误,如果查询数据库报错我们返回500错误,我们可以像下面这样写:
package mainimport ( "fmt" "github.com/pkg/errors")type Courseware struct { Id int64 Code string Name string}type ForbiddenError struct { Err error}func (e *ForbiddenError) Error() string { return "Forbidden: " + e.Err.Error()}func getCourseware(id int64) (*Courseware, error) { if id <= 0 {return nil, fmt.Errorf("invalid id: %d", id) } courseware, err := getFromDB(id) if err != nil {return nil, &ForbiddenError{err} } return courseware, nil}func getFromDB(id int64) (*Courseware, error) { return nil, errors.New("permission denied")}func main() { _, err := getCourseware(500) // 我们可以修改这里的id看下打印的结构 if err != nil {switch err := err.(type) {case *ForbiddenError:fmt.Println("500 err: ", err)default:fmt.Println("400 err: ", err)} }}
go run 9.go500 err:Forbidden: permission denied
这样看起来好像也没什么问题,现在我们稍微修改下代码,把上面ForbiddenError包装一下:package mainimport ( "fmt" "github.com/pkg/errors")type Courseware struct { Id int64 Code string Name string}type ForbiddenError struct { Err error}func (e *ForbiddenError) Error() string { return "Forbidden: " + e.Err.Error()}func getCourseware(id int64) (*Courseware, error) { if id <= 0 {return nil, fmt.Errorf("invalid id: %d", id) } courseware, err := getFromDB(id) if err != nil {return nil, fmt.Errorf("wrap err: %w", &ForbiddenError{err}) // 这里包装了一层错误 } return courseware, nil}func getFromDB(id int64) (*Courseware, error) { return nil, errors.New("permission denied")}func main() { _, err := getCourseware(500) if err != nil {switch err := err.(type) {case *ForbiddenError:fmt.Println("500 err: ", err)default:fmt.Println("400 err: ", err)} }}
go run 9.go400 err:wrap err: Forbidden: permission denied
可以看到我们的Forbidden错误进到了400里面,这并不是我们想要的结果 。之所以会这样,是因为在ForbiddenError的外面又包装了一层Error错误,使用类型断言的时候判断出来的是Error错误,所以进到了400分支 。这里我们可以使用errors.As方法,它会递归调用Unwrap方法,找到错误链中第一个与target匹配的方法:
package mainimport ( "fmt" "github.com/pkg/errors")type Courseware struct { Id int64 Code string Name string}type ForbiddenError struct { Err error}func (e *ForbiddenError) Error() string { return "Forbidden: " + e.Err.Error()}func getCourseware(id int64) (*Courseware, error) { if id <= 0 {return nil, fmt.Errorf("invalid id: %d", id) } courseware, err := getFromDB(id) if err != nil {return nil, fmt.Errorf("wrap err: %w", &ForbiddenError{err}) } return courseware, nil}func getFromDB(id int64) (*Courseware, error) { return nil, errors.New("permission denied")}func main() { _, err := getCourseware(500) if err != nil {var f *ForbiddenError // 这里实现了*ForbiddenError接口,不然会panicif errors.As(err, &f) { // 找到匹配的错误fmt.Println("500 err: ", err)} else {fmt.Println("400 err: ", err)} }}
经验总结扩展阅读
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- 十二星座中的分手大师有哪些
- 12星座因哪些错误定律葬送了爱情
- 2023年是五行金木水火土中的什么命 2023年水兔命好吗
- 世贤品如是什么电视剧中的人物?
- 我的世界怎么制作下界通道地狱门(我的世界中的下界传送门怎么制作)
- 12星座留下哪些前任的痛苦记忆
- 万嘉乐晴天是什么电视剧中的人物?
- 万大成和李元妮是什么电视剧中的人物?
- 全球首台AR电视——创维S9D电视 国货中的精品
- SLAM中的内外点