0

我来自 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中可能吗?

谢谢你。

4

1 回答 1

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"是路径模式,其他参数是请求的处理程序。该参数应该在Go 包中实现Handler接口。net/http这就是应用程序处理程序类型具有ServeHttp功能的原因。

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

我认为这将解释这种情况。如果有任何不清楚的地方,请发表评论。

于 2021-05-07T04:55:07.667 回答