之前的文章《Golang 中的 errors 包详解》详细讲解了 errors 包的主要类型和函数,以及它们的使用方法。本文结合之前讲解的知识,来讲解一下根据自己或团队的项目要求如何返回自定义的 error 类型。
在日常开发中,返回自定义 error 类型是一种很常见的做法,因为很多场景下标准库 errors 包中的 error 类型并不能满足需求。例如开发 web 项目时,常见的返回结构可能如下:
{"code":xxx, "data":yyy,"msg":"zzz"}
code 为状态码,data 为接口返回的业务数据,msg 为错误信息。针对这种响应结构,一般做法是调用处理业务逻辑的函数或方法,返回业务数据和 error,然后在 api 层拼装成这种结构。所以 error 对象至少要包含状态码 code 和错误信息 msg,这时候就需要实现自己的 error 类型了。
在 Golang 中,可以通过创建一个实现了 error 接口(该接口仅包含一个 Error 方法)的结构体来返回自定义的 error 类型。自定义的 error 类型可以包含自己定义的字段和方法,以提供更多的错误信息和上下文信息。看一个示例来展示如何创建和返回自定义 error 类型:
package main
import (
"errors"
"fmt"
)
// 自定义的错误类型
type MyError struct {
code int64
msg string
}
// 实现error接口的Error方法
func (e MyError) Error() string {
return fmt.Sprintf("Error: [%d] %s", e.code, e.msg)
}
func (e MyError) Code() int64 {
return e.code
}
func (e MyError) Msg() string {
return e.msg
}
// 函数示例,返回自定义error类型
func doSomething() error {
err := MyError{
code: 500,
msg: "Something went wrong",
}
return err
}
func main() {
err := doSomething()
if err != nil {
fmt.Println(err)
}
// 比较自定义error类型
myErr := MyError{
code: 500,
msg: "Something went wrong",
}
equal := errors.Is(err, myErr)
fmt.Println(equal) // true
}
在上述示例中,首先定义了一个自定义的 MyError 类型,包含 code 和 msg 两个字段,然后实现 error 接口要求的 Error 方法。
接下来再看一个结合 gin 框架的具体使用示例,简单示例代码如下:
package main
import (
"errors"
"fmt"
"github.com/gin-gonic/gin"
"net/http"
)
// 自定义的错误类型
type MyError struct {
code int64
msg string
}
func NewError(code int64, msg string) MyError {
return MyError{
code: code,
msg: msg,
}
}
// 实现error接口的Error方法
func (e MyError) Error() string {
return fmt.Sprintf("Error: [%d] %s", e.code, e.msg)
}
func (e MyError) GetCode() int64 {
return e.code
}
func (e MyError) GetMsg() string {
return e.msg
}
// 函数示例,返回自定义error类型
func doSomething() error {
err := MyError{
code: 500,
msg: "Something went wrong",
}
return err
}
func FromError(err error) MyError {
if err == nil {
return NewError(1, "")
}
if !errors.As(err, &MyError{}) {
return NewError(-1, "")
}
return err.(MyError)
}
func TestHandler(c *gin.Context) {
err := Logic()
if err != nil {
e := FromError(err)
c.JSON(http.StatusOK, gin.H{"code": e.GetCode(), "data": nil, "msg": e.GetMsg()})
return
}
c.JSON(http.StatusOK, gin.H{"code": 1, "data": nil, "msg": ""})
}
func Logic() error {
return NewError(-1, "something went wrong")
}
func main() {
r := gin.New()
r.GET("/test", TestHandler)
r.Run(":8080")
}
通过返回自定义的 error 类型,可以提供更多的错误信息和上下文信息,使错误处理更加灵活和准确,可以根据自己的实际需求和应用场景创建自定义错误类型。