异常捕获和处理(panic/recover)"/>
go 异常捕获和处理(panic/recover)
异常处理
Golang 没有结构化异常,使用 panic 抛出错误,recover 捕获错误。
异常的使用场景简单描述:Go中可以抛出一个panic的异常,然后在defer中通过recover捕获这个异常,然后正常处理。
panic
- 内置函数
- 假如函数F中书写了panic语句,会终止其后要执行的代码,在panic所在函数F内如果存在要执行的defer函数列表,按照defer的逆序执行
- 返回函数F的调用者G,在G中,调用函数F语句之后的代码不会执行,假如函数G中存在要执行的defer函数列表,按照defer的逆序执行
- 直到goroutine整个退出,并报告错误
recover
- 内置函数
- 用来控制一个goroutine的panicking行为,捕获panic,从而影响应用的行为
- 一般的调用建议
a). 在defer函数中,通过recever来终止一个goroutine的panicking过程,从而恢复正常代码的执行
b). 可以获取通过panic传递的error
使用
1.利用recover处理panic指令,defer 必须放在 panic 之前定义,另外 recover 只有在 defer 调用的函数中才有效。否则当panic时,recover无法捕获到panic,无法防止panic扩散。
2.recover 处理异常后,逻辑并不会恢复到 panic 那个点去
3.多个 defer 会形成 defer 栈,后定义的 defer 语句会被最先调用。
func main() {test()
}func test() {defer func() {if err := recover(); err != nil {println(err.(string)) // 将 interface{} 转型为具体类型。}}()panic("panic error!")
}
-------------------------------------panic error!
延迟调用中引发的错误,可被后续延迟调用捕获,但仅最后一个错误可被捕获。
func test() {defer func() {fmt.Println(recover())}()defer func() {panic("defer panic")}()panic("test panic")
}func main() {test()
}
---------------------------------
defer panic
实例
捕获函数 recover 只有在延迟调用内直接调用才会终止错误,否则总是返回 nil。任何未捕获的错误都会沿调用堆栈向外传递。
func test() {defer func() {fmt.Println(recover()) //有效 打印正确异常}()defer recover() //无效!defer fmt.Println(recover()) //无效! 打印nildefer func() {func() {println("defer inner") // 打印defer innerrecover() //无效!}()}()panic("test panic")
}func main() {test()
}
-------------------------------------
defer inner
<nil>
test panic
如何区别使用 panic 和 error 两种方式
惯例是:导致关键流程出现不可修复性错误的使用 panic,其他使用 error。
更多推荐
go 异常捕获和处理(panic/recover)
发布评论