如果❤️我的文章有帮助,欢迎点赞、关注。这是对我继续技术创作最大的鼓励。更多往期文章在我的个人专栏 coderdao.github.io/
试了下 Golang 里面写个 try catch
前言
Golang语言有诸多优点:静态编译、协程、堪比c语言的高性能。
但是也有一些令人发指的地方 —— 经常被人调侃 五行代码,三行错误处理
的异常错误处理方式,如下问题出现的地方
func readFile(path string) ([]byte, error) {
file, err := os.Open(path)
if err != nil {
return nil, err
}
defer file.Close()
return ioutil.ReadAll(file)
}
函数readFile
的功能是读出指定文件本身的内容并将其返回,同时当有错误发生时立即向调用方报告。
根据Golang编程范式,每个可能导致错误的函数都应该将error
作为最后一个返回值,所以Golang代码中随处都是"if err != nil"语句。 五行代码,三行错误处理
的调侃段子由此在圈内流行
能不能 try catch
我之前常用编程语言 Python 和 PHP 的时候,早已经习惯 try catch
处理异常的方式。于是对 Golang 使用流程控制 配合 panic()、recover() 一直耿耿于怀,为了以后舒服今天就来实现一下原来 try catch
异常错误处理方式
Golang 语言中拥有 recover()
函数作为宕机恢复机制
,让程序在宕机流程中的 goroutine 中恢复。
所以我们能够用 recover()
函数作为程序中是否出现的钩子
。当出现异常时,由它调起我们的异常处理函数。首先简单实现以下:
package main
import "fmt"
func try(userFn func()) {
defer func() {
if err := recover();err != nil{
fmt.Printf("程序执行发生异常: %v\n", err)
}
}()
userFn()
}
func foo(num int) {
if num < 10 {
panic("number is less than 10")
}else {
panic("number is greater than 10")
}
}
func main() {
try(func() {
foo(9)
})
try(func() {
foo(11)
})
}
go run main.go
运行:
$ go run _test/demo3/test3.go
程序执行发生异常: number is less than 10
程序执行发生异常: number is greater than 10
从上可以看出,recover()
函数完成我们让它作为钩子,的作用,调起了打印 程序执行发生异常
的信息。
加入处理异常方法
捕抓到异常还需要处理异常,而不是简单打印 程序执行发生异常
。所以还需要增加匿名方法作为异常处理方法
,改造如下:
func try(userFn func(), catchFn func(err interface{})) {
defer func() {
if err := recover();err != nil{
catchFn(err)
}
}()
userFn()
}
...
func main() {
try(func() {
foo(9)
}, func(err interface{}) {
fmt.Printf("程序执行发生异常: %v\n", err)
})
try(func() {
foo(11)
}, func(err interface{}) {
fmt.Printf("程序执行发生异常: %v\n", err)
})
}
这样就可以随心所欲传入自定义处理异常的方法
总结
以上只是自己简单实现了try catch
,其中还有很多缺陷,比如:不支持 finally
函数处理,try catch
封装过于简单。这里抛转引玉,后续接着改进