【GoLang】GoLang 错误处理 -- 使用异常的思路进行处理
go处理错误常见的方式是
err := funcReturningError()
if err != nil {
// 处理错误
}
然而因为过于繁琐而饱受诟病。下文简述另一种处理错误的写法。
这种写法最初我是从标准库里看到的,代码在 https://github.com/golang/go/blob/master/src/encoding/gob/error.go 。 简言之,就是将错误用panic抛出,然后在某个边界用defer将其转为error值。这和其他用抛异常来处理错误的语言类似。 不过上述代码并不十分通用,也没有解决最开始提出的写法繁琐的问题。 受其启发,我现在用得最多的错误处理方式是这样的 https://github.com/reusee/codes/blob/master/err/err.go 。
首先是Err结构体,定义如下
type Err struct {
Pkg, Info string
Err error
}
Pkg用于标识抛出错误的包,Info是对错误的描述。Err用于包装另一个错误,一般是当前函数所调用的函数返回的,可以实现类似java的chained exception的机制,后面再细说。
另外有一个me函数(make error),用于包装error,实现很简单不提。
然后是ce函数(check error)
func ce(err error, info string) {
if err != nil {
panic(me(err, info))
}
}
这个函数检查err参数是否为nil,如果不是,则包装出一个Err结构,然后用panic抛出。 这个就是用于替代if err != nil { … }的了。
错误用panic抛出后,必须在某个边界recover,API不应该对外暴露panic,否则会和go社区整体的理念不合,自找烦恼。 负责这个的是ct函数(catch error)
func ct(err *error) {
if p := recover(); p != nil {
if e, ok := p.(error); ok {
*err = e
} else {
panic(p)
}
}
}
因为用到了recover,所以ct只能在defer函数里调用。它首先recover(),然后看是否是error,是则将其赋值到传入的*error处,否则重新panic抛出
来看看它是如何减少代码的,以 https://blog.golang.org/errors-are-values 的一段代码为例
_, err = fd.Write(p0[a:b])
if err != nil {
return err
}
_, err = fd.Write(p1[c:d])
if err != nil {
return err
}
_, err = fd.Write(p2[e:f])
if err != nil {
return err
}
用上述机制,可以写成
defer ct(&err)
_, err = fd.Write(p0[a:b])
ce(err, "write p0")
_, err = fd.Write(p1[c:d])
ce(err, "write p1")
_, err = fd.Write(p2[e:f])
ce(err, "write p2")
代码没有那样繁琐了。
另外还有一个好处是,因为Err包装了上一个错误,所以定位错误比较容易。例如下面程序
package main
func foo() (err error) {
defer ct(&err)
ce(bar(), "call bar")
return
}
func bar() (err error) {
defer ct(&err)
ce(baz(), "call baz")
return
}
func baz() (err error) {
return me(nil, "baz")
}
func main() {
ce(foo(), "call foo")
}
paniclog是这样的
panic: foobar: call bar
foobar: call baz
foobar: baz
...
可以看出最外层的error包含了直到最内层的信息,包括包名foobar(这里只用到一个包所以体现不出),比起直接将最内层的error往上抛,要直观得多。
最后说说这种写法的缺点。首先是不论有无错误都调用recover,调用recover又要使用defer函数,所以性能会受到影响。 另外因为没有 if 语句了,做覆盖测试的话,区分不出两种case了。 所以这种写法并不适合所有场景。需要压榨性能时不用,需要做覆盖测试时不用。 适合的场景是对性能要求不高的,对正确性要求也不高的。 我会用在经常变的应用代码,或者百几十行的小程序,或者测试代码里。 基础的包,还是好好写 if err != nil { … } 吧。
以上ct、me、ce等函数都不是public的,因为我使用时,是用代码生成工具复制出来用的,不需要public。 用的代码生成工具是 https://github.com/reusee/ccg ,可能会有另外一篇博文说说这个。
参考资料:
http://reusee.github.io/post/error-handling/
【GoLang】GoLang 错误处理 -- 使用异常的思路进行处理的更多相关文章
- golang 错误处理与异常
原文地址 golang 中的错误处理的哲学和 C 语言一样,函数通过返回错误类型(error)或者 bool 类型(不需要区分多种错误状态时)表明函数的执行结果,调用检查返回的错误类型值是否是 nil ...
- ASP.NET全局错误处理和异常日志记录以及IIS配置自定义错误页面
应用场景和使用目的 很多时候,我们在访问页面的时候,由于程序异常.系统崩溃会导致出现黄页.在通常的情况下,黄页对于我们来说,帮助是极大的,因为它可以帮助我们知道问题根源,甚至是哪一行代码出现了错误.但 ...
- spring boot: GlobalDefaultExceptionHandler方法内的友好错误提示,全局异常捕获
spring boot: GlobalDefaultExceptionHandler方法内的友好错误提示,全局异常捕获 当你的某个控制器内的某个方法报错,基本上回显示出java错误代码,非常不友好,这 ...
- OpenJDK源码研究笔记(一)-参数检查&抛出带关键错误提示信息的异常
OpenJDK源码研究笔记系列文章,是我在阅读OpenJDK7源码的过程中的一些体会.收获.看法. 把研究过程中的成长和收获一点点地整理出来,是对自己研究学习的一个小结,也有可能给学习Java的一些同 ...
- [golang]golang如何覆盖输出console,实现进度条;golang一个骚气的进度提示库
[golang]golang如何覆盖输出console,实现进度条 package main import( "fmt" "os" "time&quo ...
- Go_19: Golang 中错误与异常需要重新认识
如何进行错误处理,这是一个Go程序员之间,特别是一些新的Go程序员,会经常讨论的问题.讨论到最后往往由于以下代码的多次出现而变成了抱怨. if err != nil { return err } 我们 ...
- Golang 中错误与异常需要重新认识
如何进行错误处理,这是一个Go程序员之间,特别是一些新的Go程序员,会经常讨论的问题.讨论到最后往往由于以下代码的多次出现而变成了抱怨. if err != nil { return err } 我们 ...
- [Golang]-4 错误处理、Panic、Defer
目录 错误和异常 案例 Panic Defer 使用 defer+recover 来处理错误 参考文章: Go 语言使用一个独立的·明确的返回值来传递错误信息的.这与使用异常的 Java 和 Ruby ...
- 个人犯的一个golang routine错误
这个其实不是错误,2个写法没有区别.-2015.11.22 认识golang也不少时间了,也做过几个项目.最近发现之前用golang写的一个服务,内存涨得比较快,一直没找出来原因来.今天把疑惑发到群里 ...
随机推荐
- LaTeX 页眉页脚的设置
Latex中页眉页脚的设置 1. 首先要加页眉页脚的话,需要启动宏: 我通常用fancyhdr宏包来设置页眉和页脚. \usepackage{fancyhdr} 我们在 LaTeX 中先把 page ...
- Outlook2003收到的邮件不能显示图片,但转发或回复可以 故障排错
转自:http://wenku.baidu.com/link?url=4O0Wpsnk_OasA7B-M4KMJ4H-WnLicT4vfVOT7AvXhTjpn-p2gBTNsXR4W230r0gAE ...
- MySQL笔记-最简单的方法来解决找不到mysqld.sock文件的问题
首先,环境:ubuntu 14.04,采用apt-get的方式安装的,手动安装可能路径设置稍有区别. 1.安装MySQL后,用命令行首次启动时发现找不到Mysqld.sock文件,提示: ERROR ...
- 在VS2013中强制IIS Express应用程序池使用经典模式
直接在文件夹地址栏输入 %userprofile%\documents\iisexpress\config\applicationhost.config 会打开上边的配置文件 将 <siteDe ...
- CSS实现底部固定
html代码结构 <body> <div class="wrap"> <div class="head"></div& ...
- 合并两个结构完全相同的DataTable
两个结构一模一样的DataTable如何合并? 例子:使用Winform进行演示,表2的数据为固定的,表1的数据可以动态添加,通过合并按钮合并表1和表2的数据到表3 1.规定公共的DataTable结 ...
- maven jar包库
如果你的项目不是maven项目,比如ant,你的项目需要某些jar包的时候可以到maven 的jar包中心库下载 地址:http://search.maven.org/ http://mvnrepos ...
- 昨天的这个先补上--这个是关于 JQ 的移动 和 渐变特效的点击事件
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...
- MySQL中MAX函数与Group By一起使用的注意事项(转)
mysql> select * from test; +----+-------+------+-------+ | id | name | age | class | +----+------ ...
- vim中添加molokai.vim 配色安装
无意中发现知乎中讨论的话题: 你认为最好看的 Vim 配色方案(color scheme)是哪款? 网友回答 排在第一位的是:molokai 啊,最经典的配色 既然molokai这么经典,当然要用了. ...