1. 异常处理

Golang 没有结构化异常,使用 panic 抛出错误,recover 捕获错误。

异常的使用场景简单描述:Go中可以抛出一个panic的异常,然后在defer中通过recover捕获这个异常,然后正常处理。

panic:

  1. 1、内置函数
  2. 2、假如函数F中书写了panic语句,会终止其后要执行的代码,在panic所在函数F内如果存在要执行的defer函数列表,按照defer的逆序执行
  3. 3、返回函数F的调用者G,在G中,调用函数F语句之后的代码不会执行,假如函数G中存在要执行的defer函数列表,按照defer的逆序执行
  4. 4、直到goroutine整个退出,并报告错误

recover:

  1. 1、内置函数
  2. 2、用来控制一个goroutinepanicking行为,捕获panic,从而影响应用的行为
  3. 3、一般的调用建议
  4. a). defer函数中,通过recever来终止一个goroutinepanicking过程,从而恢复正常代码的执行
  5. b). 可以获取通过panic传递的error

注意:

  1. 1.利用recover处理panic指令,defer 必须放在 panic 之前定义,另外 recover 只有在 defer 调用的函数中才有效。否则当panic时,recover无法捕获到panic,无法防止panic扩散。
  2. 2.recover 处理异常后,逻辑并不会恢复到 panic 那个点去,函数跑到 defer 之后的那个点。
  3. 3.多个 defer 会形成 defer 栈,后定义的 defer 语句会被最先调用。
  1. package main
  2. func main() {
  3. test()
  4. }
  5. func test() {
  6. defer func() {
  7. if err := recover(); err != nil {
  8. println(err.(string)) // 将 interface{} 转型为具体类型。
  9. }
  10. }()
  11. panic("panic error!")
  12. }

输出结果:

  1. panic error!

由于 panic、recover 参数类型为 interface{},因此可抛出任何类型对象。

  1. func panic(v interface{})
  2. func recover() interface{}

向已关闭的通道发送数据会引发panic

  1. package main
  2. import (
  3. "fmt"
  4. )
  5. func main() {
  6. defer func() {
  7. if err := recover(); err != nil {
  8. fmt.Println(err)
  9. }
  10. }()
  11. var ch chan int = make(chan int, 10)
  12. close(ch)
  13. ch <- 1
  14. }

输出结果:

  1. send on closed channel

延迟调用中引发的错误,可被后续延迟调用捕获,但仅最后一个错误可被捕获。

  1. package main
  2. import "fmt"
  3. func test() {
  4. defer func() {
  5. fmt.Println(recover())
  6. }()
  7. defer func() {
  8. panic("defer panic")
  9. }()
  10. panic("test panic")
  11. }
  12. func main() {
  13. test()
  14. }

输出:

  1. defer panic

捕获函数 recover 只有在延迟调用内直接调用才会终止错误,否则总是返回 nil。任何未捕获的错误都会沿调用堆栈向外传递。

  1. package main
  2. import "fmt"
  3. func test() {
  4. defer func() {
  5. fmt.Println(recover()) //有效
  6. }()
  7. defer recover() //无效!
  8. defer fmt.Println(recover()) //无效!
  9. defer func() {
  10. func() {
  11. println("defer inner")
  12. recover() //无效!
  13. }()
  14. }()
  15. panic("test panic")
  16. }
  17. func main() {
  18. test()
  19. }

输出:

  1. defer inner
  2. <nil>
  3. test panic

使用延迟匿名函数或下面这样都是有效的。

  1. package main
  2. import (
  3. "fmt"
  4. )
  5. func except() {
  6. fmt.Println(recover())
  7. }
  8. func test() {
  9. defer except()
  10. panic("test panic")
  11. }
  12. func main() {
  13. test()
  14. }

输出结果:

  1. test panic

如果需要保护代码 段,可将代码块重构成匿名函数,如此可确保后续代码被执 。

  1. package main
  2. import "fmt"
  3. func test(x, y int) {
  4. var z int
  5. func() {
  6. defer func() {
  7. if recover() != nil {
  8. z = 0
  9. }
  10. }()
  11. panic("test panic")
  12. z = x / y
  13. return
  14. }()
  15. fmt.Printf("x / y = %d\n", z)
  16. }
  17. func main() {
  18. test(2, 1)
  19. }

输出结果:

  1. x / y = 0

除用 panic 引发中断性错误外,还可返回 error 类型错误对象来表示函数调用状态。

  1. type error interface {
  2. Error() string
  3. }

标准库 errors.New 和 fmt.Errorf 函数用于创建实现 error 接口的错误对象。通过判断错误对象实例来确定具体错误类型。

  1. package main
  2. import (
  3. "errors"
  4. "fmt"
  5. )
  6. var ErrDivByZero = errors.New("division by zero")
  7. func div(x, y int) (int, error) {
  8. if y == 0 {
  9. return 0, ErrDivByZero
  10. }
  11. return x / y, nil
  12. }
  13. func main() {
  14. defer func() {
  15. fmt.Println(recover())
  16. }()
  17. switch z, err := div(10, 0); err {
  18. case nil:
  19. println(z)
  20. case ErrDivByZero:
  21. panic(err)
  22. }
  23. }

输出结果:

  1. division by zero

Go实现类似 try catch 的异常处理

  1. package main
  2. import "fmt"
  3. func Try(fun func(), handler func(interface{})) {
  4. defer func() {
  5. if err := recover(); err != nil {
  6. handler(err)
  7. }
  8. }()
  9. fun()
  10. }
  11. func main() {
  12. Try(func() {
  13. panic("test panic")
  14. }, func(err interface{}) {
  15. fmt.Println(err)
  16. })
  17. }

输出结果:

  1. test panic

如何区别使用 panic 和 error 两种方式?

惯例是:导致关键流程出现不可修复性错误的使用 panic,其他使用 error。