在c++中,我们对于异常的处理通常有两种形式,一种是返回指定错误码,一种throw出来,由外层的try...catch进行捕获。在go中,通常用error类型来返回我们期望的错误,例如:
f, err := os.Open("filename.ext")
if err != nil {
log.Fatal(err)
}
调用Open函数后,返回的error能够帮我们判断文件是否成功打开。当然这么写没什么问题,但试想下,在一个庞大的项目中,用上述方法进行错误判断,那么代码的冗余度就会很高,本文搜集了网上的一些资料,希望对error的处理能够更简洁些。
异常的抛出
在go中,错误用error类型表示,这是一个接口,接口的声明如下:
type error interface {
Error() string
}
以下为返回一段error的例子:
func main() {
f,err := Sqrt(-99)
fmt.Print(err)
}
func Sqrt(f float64) (float64, error) {
if f < 0 {
return 0, errors.New("math: square root of negative number")
}
// implementation
}
error.New()会返回一个error类型,我们可以将error用print输出,但实际工程中,error.New还不是最常用的,因为我们希望将一些重要参数也打印在error中,那么可以使用fmt.Errorf(),如下:
if f < 0 {
return 0, fmt.Errorf("math: square root of negative number %g", f)
}
如果你的工程需要频繁返回某个类对象的error,那么我们可以自己定义这个类的Error函数,因为刚刚也说了,error是一个接口,那就意味着我们可以实现自己期望的error函数。如下,我们定义了一个float64类型的Error函数,。
type NegativeSqrtError float64
func (f NegativeSqrtError) Error() string {
return fmt.Sprintf("math: square root of negative number %g", float64(f))
}
简洁的错误处理方式
func init() {
http.HandleFunc("/view", viewRecord)
}
func viewRecord(w http.ResponseWriter, r *http.Request) {
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 {
http.Error(w, err.Error(), 500)
return
}
if err := viewTemplate.Execute(w, record); err != nil {
http.Error(w, err.Error(), 500)
}
}
上述代码已经比文章刚开头处理error的形式好多了,使用如下形式:
err:=func(....);err!=nil{...}
但是异常流多了,还是会有大量的冗余代码,那我们干脆让函数返回一个error类型,如下:
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)
}
这段代码比上面那段更加清爽了些。但如果再有一个函数调用了viewRecord,当最上层打印出error,我们并不知道是到底是哪个函数出了问题,所以为了方便查看,最好返回的时候打一些日志,或者打不同的错误信息,针对返回的error内容,可以采用以下几种方式:
- 重新实现该类的error接口(刚刚上面有说);
- 用一个结构体,结构体中包括error,errCode,errMsg等基本的错误信息,函数返回结构体;
- 使用https://github.com/pkg/errors,这个库可以允许你在返回的error信息中加一些错误信息。
个人感觉第三种方法还方便些,
参考文章:
https://hackernoon.com/golang-handling-errors-gracefully-8e27f1db729f