learn-tech/专栏/22讲通关Go语言-完/07错误处理:如何通过error、deferred、panic等处理错误?.md
2024-10-15 23:13:09 +08:00

341 lines
13 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

因收到Google相关通知网站将会择期关闭。相关通知内容
07 错误处理:如何通过 error、deferred、panic 等处理错误?
上节课我为你讲解了结构体和接口,并留了一个小作业,让你自己练习实现有两个方法的接口。现在我就以“人既会走也会跑”为例进行讲解。
首先定义一个接口 WalkRun它有两个方法 Walk 和 Run如下面的代码所示
type WalkRun interface {
Walk()
Run()
}
现在就可以让结构体 person 实现这个接口了,如下所示:
func (p *person) Walk(){
fmt.Printf("%s能走\n",p.name)
}
func (p *person) Run(){
fmt.Printf("%s能跑\n",p.name)
}
关键点在于,让接口的每个方法都实现,也就实现了这个接口。
提示:%s 是占位符,和 p.name 对应,也就是 p.name 的值,具体可以参考 fmt.Printf 函数的文档。
下面进行本节课的讲解。这节课我会带你学习 Go 语言的错误和异常,在我们编写程序的时候,可能会遇到一些问题,该怎么处理它们呢?
错误
在 Go 语言中,错误是可以预期的,并且不是非常严重,不会影响程序的运行。对于这类问题,可以用返回错误给调用者的方法,让调用者自己决定如何处理。
error 接口
在 Go 语言中,错误是通过内置的 error 接口表示的。它非常简单,只有一个 Error 方法用来返回具体的错误信息,如下面的代码所示:
type error interface {
Error() string
}
在下面的代码中,我演示了一个字符串转整数的例子:
ch07/main.go
func main() {
i,err:=strconv.Atoi("a")
if err!=nil {
fmt.Println(err)
}else {
fmt.Println(i)
}
}
这里我故意使用了字符串 “a”尝试把它转为整数。我们知道 “a” 是无法转为数字的,所以运行这段程序,会打印出如下错误信息:
strconv.Atoi: parsing "a": invalid syntax
这个错误信息就是通过接口 error 返回的。我们来看关于函数 strconv.Atoi 的定义,如下所示:
func Atoi(s string) (int, error)
一般而言error 接口用于当方法或者函数执行遇到错误时进行返回,而且是第二个返回值。通过这种方式,可以让调用者自己根据错误信息决定如何进行下一步处理。
小提示:因为方法和函数基本上差不多,区别只在于有无接收者,所以以后当我称方法或函数,表达的是一个意思,不会把这两个名字都写出来。
error 工厂函数
除了可以使用其他函数,自己定义的函数也可以返回错误信息给调用者,如下面的代码所示:
ch07/main.go
func add(a,b int) (int,error){
if a<0 || b<0 {
return 0,errors.New("a或者b不能为负数")
}else {
return a+b,nil
}
}
add 函数会在 a 或者 b 任何一个为负数的情况下返回一个错误信息如果 ab 都不为负数错误信息部分会返回 nil这也是常见的做法所以调用者可以通过错误信息是否为 nil 进行判断
下面的 add 函数示例是使用 errors.New 这个工厂函数生成的错误信息它接收一个字符串参数返回一个 error 接口这些在上节课的结构体和接口部分有过详细介绍不再赘述
ch07/main.go
sum,err:=add(-1,2)
if err!=nil {
fmt.Println(err)
}else {
fmt.Println(sum)
}
自定义 error
你可能会想上面采用工厂返回错误信息的方式只能传递一个字符串也就是携带的信息只有字符串如果想要携带更多信息比如错误码信息该怎么办呢这个时候就需要自定义 error
自定义 error 其实就是先自定义一个新类型比如结构体然后让这个类型实现 error 接口如下面的代码所示
ch07/main.go
type commonError struct {
errorCode int //错误码
errorMsg string //错误信息
}
func (ce *commonError) Error() string{
return ce.errorMsg
}
有了自定义的 error就可以使用它携带更多的信息现在我改造上面的例子返回刚刚自定义的 commonError如下所示
ch07/main.go
return 0, &commonError{
errorCode: 1,
errorMsg: "a或者b不能为负数"}
我通过字面量的方式创建一个 *commonError 返回其中 errorCode 值为 1errorMsg 值为 a 或者 b 不能为负数”。
error 断言
有了自定义的 error并且携带了更多的错误信息后就可以使用这些信息了你需要先把返回的 error 接口转换为自定义的错误类型用到的知识是上节课的类型断言
下面代码中的 err.(*commonError) 就是类型断言在 error 接口上的应用也可以称为 error 断言
ch07/main.go
sum, err := add(-1, 2)
if cm,ok:=err.(*commonError);ok{
fmt.Println("错误代码为:",cm.errorCode,"错误信息为",cm.errorMsg)
} else {
fmt.Println(sum)
}
如果返回的 ok true说明 error 断言成功正确返回了 *commonError 类型的变量 cm所以就可以像示例中一样使用变量 cm errorCode errorMsg 字段信息了
错误嵌套
Error Wrapping
error 接口虽然比较简洁但是功能也比较弱想象一下假如我们有这样的需求基于一个存在的 error 再生成一个 error需要怎么做呢这就是错误嵌套
这种需求是存在的比如调用一个函数返回了一个错误信息 error在不想丢失这个 error 的情况下又想添加一些额外信息返回新的 error这时候我们首先想到的应该是自定义一个 struct如下面的代码所示
type MyError struct {
err error
msg string
}
这个结构体有两个字段其中 error 类型的 err 字段用于存放已存在的 errorstring 类型的 msg 字段用于存放新的错误信息这种方式就是 error 的嵌套
现在让 MyError 这个 struct 实现 error 接口然后在初始化 MyError 的时候传递存在的 error 和新的错误信息如下面的代码所示
func (e *MyError) Error() string {
return e.err.Error() + e.msg
}
func main() {
//err是一个存在的错误可以从另外一个函数返回
newErr := MyError{err, "数据上传问题"}
}
这种方式可以满足我们的需求但是非常烦琐因为既要定义新的类型还要实现 error 接口所以从 Go 语言 1.13 版本开始Go 标准库新增了 Error Wrapping 功能让我们可以基于一个存在的 error 生成新的 error并且可以保留原 error 信息如下面的代码所示
ch07/main.go
e := errors.New("原始错误e")
w := fmt.Errorf("Wrap了一个错误:%w", e)
fmt.Println(w)
Go 语言没有提供 Wrap 函数而是扩展了 fmt.Errorf 函数然后加了一个 %w通过这种方式便可以生成 wrapping error
errors.Unwrap 函数
既然 error 可以包裹嵌套生成一个新的 error那么也可以被解开即通过 errors.Unwrap 函数得到被嵌套的 error
Go 语言提供了 errors.Unwrap 用于获取被嵌套的 error比如以上例子中的错误变量 w 就可以对它进行 unwrap获取被嵌套的原始错误 e
下面我们运行以下代码
fmt.Println(errors.Unwrap(w))
可以看到这样的信息原始错误 e”。
原始错误e
errors.Is 函数
有了 Error Wrapping 你会发现原来用的判断两个 error 是不是同一个 error 的方法失效了比如 Go 语言标准库经常用到的如下代码中的方式
if err == os.ErrExist
为什么会出现这种情况呢由于 Go 语言的 Error Wrapping 功能令人不知道返回的 err 是否被嵌套又嵌套了几层
于是 Go 语言为我们提供了 errors.Is 函数用来判断两个 error 是否是同一个如下所示
func Is(err, target error) bool
以上就是errors.Is 函数的定义可以解释为
如果 err target 是同一个那么返回 true
如果 err 是一个 wrapping errortarget 也包含在这个嵌套 error 链中的话也返回 true
可以简单地概括为两个 error 相等或 err 包含 target 的情况下返回 true其余返回 false我们可以用上面的示例判断错误 w 中是否包含错误 e试试运行下面的代码来看打印的结果是不是 true
fmt.Println(errors.Is(w,e))
errors.As 函数
同样的原因有了 error 嵌套后error 断言也不能用了因为你不知道一个 error 是否被嵌套又嵌套了几层所以 Go 语言为解决这个问题提供了 errors.As 函数比如前面 error 断言的例子可以使用 errors.As 函数重写效果是一样的如下面的代码所示
ch07/main.go
var cm *commonError
if errors.As(err,&cm){
fmt.Println("错误代码为:",cm.errorCode,"错误信息为",cm.errorMsg)
} else {
fmt.Println(sum)
}
所以在 Go 语言提供的 Error Wrapping 能力下我们写的代码要尽可能地使用 IsAs 这些函数做判断和转换
Deferred 函数
在一个自定义函数中你打开了一个文件然后需要关闭它以释放资源不管你的代码执行了多少分支是否出现了错误文件是一定要关闭的这样才能保证资源的释放
如果这个事情由开发人员来做随着业务逻辑的复杂会变得非常麻烦而且还有可能会忘记关闭基于这种情况Go 语言为我们提供了 defer 函数可以保证文件关闭后一定会被执行不管你自定义的函数出现异常还是错误
下面的代码是 Go 语言标准包 ioutil 中的 ReadFile 函数它需要打开一个文件然后通过 defer 关键字确保在 ReadFile 函数执行结束后f.Close() 方法被执行这样文件的资源才一定会释放
func ReadFile(filename string) ([]byte, error) {
f, err := os.Open(filename)
if err != nil {
return nil, err
}
defer f.Close()
//省略无关代码
return readAll(f, n)
}
defer 关键字用于修饰一个函数或者方法使得该函数或者方法在返回前才会执行也就说被延迟但又可以保证一定会执行
以上面的 ReadFile 函数为例 defer 修饰的 f.Close 方法延迟执行也就是说会先执行 readAll(f, n)然后在整个 ReadFile 函数 return 之前执行 f.Close 方法
defer 语句常被用于成对的操作如文件的打开和关闭加锁和释放锁连接的建立和断开等不管多么复杂的操作都可以保证资源被正确地释放
Panic 异常
Go 语言是一门静态的强类型语言很多问题都尽可能地在编译时捕获但是有一些只能在运行时检查比如数组越界访问不相同的类型强制转换等这类运行时的问题会引起 panic 异常
除了运行时可以产生 panic 我们自己也可以抛出 panic 异常假设我需要连接 MySQL 数据库可以写一个连接 MySQL 的函数connectMySQL如下面的代码所示
ch07/main.go
func connectMySQL(ip,username,password string){
if ip =="" {
panic("ip不能为空")
}
//省略其他代码
}
connectMySQL 函数中如果 ip 为空会直接抛出 panic 异常这种逻辑是正确的因为数据库无法连接成功的话整个程序运行起来也没有意义所以就抛出 panic 终止程序的运行
panic Go 语言内置的函数可以接受 interface{} 类型的参数也就是任何类型的值都可以传递给 panic 函数如下所示
func panic(v interface{})
小提示interface{} 是空接口的意思 Go 语言中代表任意类型
panic 异常是一种非常严重的情况会让程序中断运行使程序崩溃所以如果是不影响程序运行的错误不要使用 panic使用普通错误 error 即可
Recover 捕获 Panic 异常
通常情况下我们不对 panic 异常做任何处理因为既然它是影响程序运行的异常就让它直接崩溃即可但是也的确有一些特例比如在程序崩溃前做一些资源释放的处理这时候就需要从 panic 异常中恢复才能完成处理
Go 语言中可以通过内置的 recover 函数恢复 panic 异常因为在程序 panic 异常崩溃的时候只有被 defer 修饰的函数才能被执行所以 recover 函数要结合 defer 关键字使用才能生效
下面的示例是通过 defer 关键字 + 匿名函数 + recover 函数从 panic 异常中恢复的方式
ch07/main.go
func main() {
defer func() {
if p:=recover();p!=nil{
fmt.Println(p)
}
}()
connectMySQL("","root","123456")
}
运行这个代码可以看到如下的打印输出这证明 recover 函数成功捕获了 panic 异常
ip 不能为空
通过这个输出的结果也可以发现recover 函数返回的值就是通过 panic 函数传递的参数值
总结
这节课主要讲了 Go 语言的错误处理机制包括 errordeferpanic errorpanic 这两种错误机制中Go 语言更提倡 error 这种轻量错误而不是 panic