【问题标题】:Go Deep understanding of custom error handling in REST APIGo 深入理解 REST API 中的自定义错误处理
【发布时间】:2021-05-07 04:05:17
【问题描述】:

我来自 Node.js 的世界,在那里我从使用 Express 构建 Web 应用程序中学到了很多东西,我认为在处理错误和如果出现意外错误,有一种很棒的方法可以捕获它。

所以,我在 Go 中寻找同样的东西。我不知道我是否已经找到它,但是我发现的内容来自此链接https://astaxie.gitbooks.io/build-web-application-with-golang/content/en/11.1.html 以及我阅读的一些文章,似乎许多开发人员都在使用相同的方法。

我担心的是,我不理解下面这段代码的某些部分。

type appHandler func(http.ResponseWriter, *http.Request) error

func (fn appHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    if err := fn(w, r); err != nil {
        http.Error(w, err.Error(), 500)
    }
}

我知道可以在 go 中创建自定义类型,但严重的是我不明白这个是什么意思,也不明白如何在 http.Serve

中理解它
type appHandler func(http.ResponseWriter, *http.Request) error

我没有注意到的一件事是

func (fn AppHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {

在我阅读的大部分代码中,一般来说,它是一个结构或替换类型(我的意思是type Account int),但在这里,它是一个函数。 我想了解它如何帮助处理错误。

ServeHTTP 的实现内部,我们有这行err := fn(w, r)。 请问,你能解释一下吗?

在同一篇文章中,我们有这段代码

func viewRecord(w http.ResponseWriter, r *http.Request) error {
    c := appengine.NewContext(r)
    key := datastore.NewKey(c, "Record", r.FormValue("id"), 0, nil)
    record := new(Record)
    if err := datastore.Get(c, key, record); err != nil {
        return err
    }
    return viewTemplate.Execute(w, record)
}

还有这一行

func init() {
    http.Handle("/view", appHandler(viewRecord))
}

拜托,你能帮我理解这个appHandler(viewRecord)吗?究竟是什么?这是一个实例化,是铸造吗? 它应该做什么?我的意思是,如何理解似乎很关键的那条线?

最后一个问题,请。是否有可能在处理请求的过程中捕捉到任何地方可能发生的错误?在 Node.js 中,您可以执行类似的操作

const app = express()

const errorHandler = (
  err: Error,
  req: Request,
  res: Response,
  next: NextFunction
) => {
  if (err instanceof CustomError) {
    return res.status(err.statusCode).send({ errors: err.serializeErrors() });
  }
  
  res.status(500).send({ 
    errors: [{ message: err.message || 'Something went wrong' }]
  });
}

app.use(errorHandler())

类似的东西,在 Go 中可以吗?

谢谢。

【问题讨论】:

    标签: node.js rest express go go-http


    【解决方案1】:

    在 Go 中,我们可以从任何原始类型定义自定义类型, 示例:

    type A int
    type Person struct{}
    

    这对于函数来说也很常见。您可以定义具有所需函数签名的 Go 类型,如上面示例中的 appHandler 类型。我在下面添加了简单的示例。

    package main
    
    import (
        "errors"
        "fmt"
    )
    
    type IntFun func(a,b int) error
    func main(){
        var addFunc IntFun
        addFunc = func(a, b int) error {
            if a ==0 || b == 0 {
                return errors.New(`zero valued inputs`)
            }
            fmt.Println(`sum := `, a+b)
            return nil
        }
    
        addFunc.Add(5, 3) //Output: sum :=  8
        addFunc.Add(0, 0) //Output: zero valued inputs
    }
    
    func (fn IntFun) Add(a, b int)  {
        err := fn(a,b)
        if err != nil {
            fmt.Println(err)
            return
        }
    }
    

    也可以参考user-defined-function-type-go

    在你的情况下,

    func init() {
        http.Handle("/view", appHandler(viewRecord))
    }
    

    这是 http 端点处理部分。 "/view" 是路径模式,其他参数是请求的处理程序。该参数应该在net/http Go 包中实现Handler 接口。这就是为什么应用处理程序类型具有ServeHttp 函数的原因。

    viewRecord 函数也兼容appHandler 函数类型。它处理到达“/view”端点的请求。所以它被强制转换为appHandler 类型并传递给http.Handle("/view", appHandler(viewRecord)) 句柄函数。

    我认为这将解释这种情况。如果有什么不清楚的地方请评论。

    【讨论】:

    • 语言名称是Go,不是golang。
    • 对不起@Volker。我已经纠正了。感谢您的评论。
    猜你喜欢
    • 2020-10-26
    • 2013-07-02
    • 1970-01-01
    • 1970-01-01
    • 2010-12-26
    • 1970-01-01
    • 2014-06-16
    相关资源
    最近更新 更多