Golang 恢复程序
优质
小牛编辑
147浏览
2023-12-01
程序和人一样都需要具备一定的容错能力,学会知错就改,所以如果不是不可恢复性、导致系统无法正常工作的错误,如果发生了 panic 我们需要恢复程序,让程序继续执行,并且需要记录到底犯了什么错误
在 Go 语言中我们可以通过 defer 和 recover 来实现 panic 异常的捕获,让程序继续执行
package main import "fmt" func div(a, b int) (res int) { // 定义一个延迟调用的函数, 用于捕获panic异常 // 注意: 一定要在panic之前定义 defer func() { if err := recover(); err != nil{ res = -1 fmt.Println(err) // 除数不能为0 } }() if(b == 0){ //err = errors.New("除数不能为0") panic("除数不能为0") }else{ res = a / b } return } func setValue(arr []int, index int ,value int) { arr[index] = value } func main() { res := div(10, 0) fmt.Println(res) // -1 }
panic 注意点
panic 异常会沿着调用堆栈向外传递,所以也可以在外层捕获
package main import "fmt" func div(a, b int) (res int) { if(b == 0){ //err = errors.New("除数不能为0") panic("除数不能为0") }else{ res = a / b } return } func main() { // panic异常会沿着调用堆栈向外传递, 所以也可以在外层捕获 defer func() { if err := recover(); err != nil{ fmt.Println(err) // 除数不能为0 } }() div(10, 0) }
多个异常,只有第一个会被捕获
package main import "fmt" func test1() { // 多个异常,只有第一个会被捕获 defer func() { if err := recover(); err != nil{ fmt.Println(err) // 异常A } }() panic("异常A") // 相当于return, 后面代码不会继续执行 panic("异常B") } func main() { test1(10, 0) }
如果有异常写在 defer 中,那么只有 defer 中的异常会被捕获
package main import "fmt" func test2() { // 如果有异常写在defer中, 并且其它异常写在defer后面, 那么只有defer中的异常会被捕获 defer func() { if err := recover(); err != nil{ fmt.Println(err) // 异常A } }() defer func() { panic("异常B") }() panic("异常A") } func main() { test1(10, 0) }