问题描述
我来自 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"),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 中可以实现吗?
谢谢。
解决方法
在 Go 中,我们可以从任何原始类型定义自定义类型, 例子:
type A int
type Person struct{}
这对于函数来说很常见。您可以像上面示例中的 appHandler
类型一样定义具有所需函数签名的 Go 类型。我在下面添加了简单的例子。
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
函数也兼容 type appHandler
函数类型。它处理到达“/view”端点的请求。因此它被强制转换为 appHandler
类型并传递给 http.Handle("/view",appHandler(viewRecord))
句柄函数。
我认为这将解释这种情况。如果有什么不清楚的,请评论。