29.Go异常处理-recover

2021-09-17 10:19:39 浏览数 (1)

29.Go异常处理-recover

4. recover

运行时panic异常一旦被引发就会导致程序崩溃。这当然不是我们愿意看到的,因为谁也不能保证程序不会发生任何运行时错误。

Go语言为我们提供了专用于“拦截”运行时panic的内建函数——recover。它可以是当前的程序从运行时panic的状态中恢复并重新获得流程控制权。

语法如下:

代码语言:javascript复制
func recover() interface{} 

注意:recover只有在defer调用的函数中有效。

使用 recover() 函数阻止 panic 导致的程序崩溃,示例如下:

代码语言:javascript复制
package main

import "fmt"

func TestA()  {
   fmt.Println("func TestA()")
}

func TestB(x int)  {
   // 设置recover, 防止程序崩溃
   defer func() {
      recover()
   }()

   var a [10]int
   a[x] = 111 // 当x为20时候,导致数组越界,产生一个panic, 导致程序崩溃
}

func TestC()  {
   fmt.Println("func TestC()")
}

func main() {
   TestA()
   TestB(11) // TestB()发生异常,中断程序
   TestC()
} 

以上程序的运行结果如下:

代码语言:javascript复制
func TestA()
func TestC() 

通过以上程序,我们发现虽然TestB( )函数会导致整个应用程序崩溃,但是由于在改函数中调用了recover( )函数,所以整个函数并没有崩溃。虽然程序没有崩溃,但是我们也没有看到任何的提示信息,那么怎样才能够看到相应的提示信息呢?

使用 recover() 返回崩溃的错误信息

可以直接打印recover( )函数的返回结果,如下所示:

代码语言:javascript复制
package main

import "fmt"

func TestA()  {
   fmt.Println("func TestA()")
}

func TestB(x int)  {
   // 设置recover, 防止程序崩溃
   defer func() {
      //recover()
      //可以打印panic的错误信息
      fmt.Println(recover())
   }()

   var a [10]int
   a[x] = 111 // 当x为20时候,导致数组越界,产生一个panic, 导致程序崩溃
}

func TestC()  {
   fmt.Println("func TestC()")
}

func main() {
   TestA()
   TestB(11) // TestB()发生异常,中断程序
   TestC()
} 

输出结果如下:

从输出结果发现,确实打印出了相应的错误信息。

但是,如果程序没有出错,也就是数组下标没有越界,会出现什么情况呢?

image-20210606002619770

这时输出的是空,但是我们希望程序没有错误的时候,不输出任何内容。

所以,程序修改如下:

image-20210606002749765

代码语言:javascript复制
package main

import "fmt"

func TestA()  {
   fmt.Println("func TestA()")
}

func TestB(x int)  {
   // 设置recover, 防止程序崩溃
   defer func() {
      //recover()
      //可以打印panic的错误信息
      if err := recover(); err != nil{
         fmt.Println(recover())
      }
   }()

   var a [10]int
   a[x] = 111 // 当x为20时候,导致数组越界,产生一个panic, 导致程序崩溃
}

func TestC()  {
   fmt.Println("func TestC()")
}

func main() {
   TestA()
   TestB(1) // TestB()发生异常,中断程序
   TestC()
}

通过以上代码,发现其实就是加了一层判断。

0 人点赞