Golang中的panic与recover怎么使用

其他教程   发布日期:2025年04月23日   浏览次数:165

这篇文章主要介绍了Golang中的panic与recover怎么使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Golang中的panic与recover怎么使用文章都会有所收获,下面我们一起来看看吧。

panic

panic作用是终止当前正在运行的程序(包括所有协程)并输出导致异常的堆栈信息。在遇到无法处理的异常情况时,例如比如数组越界、操作未初始化的map、空指针等都会触发panic。主动触发panic示例:

  1. package main
  2. func main() {
  3. // 未处理的自定义异常
  4. customException := "an error occurred"
  5. panic(customException)
  6. }

会输出如下信息:

panic: an error occurred

goroutine 1 [running]:
main.main()
/Users/ning/projects/go/workspace/hello/panic/main.go:7 +0x34

Process finished with the exit code 2

数据越界导致panic示例:

  1. package main
  2. import "fmt"
  3. func main() {
  4. a := [2]int{4, 5}
  5. fmt.Println(a[3])
  6. }

会输出如下信息:

# command-line-arguments
./main.go:11:16: invalid argument: array index 3 out of bounds [0:2]

Compilation finished with exit code 2

recover

recover可以让触发了panic的程序继续运行,recover仅在延迟函数defer中有效,在正常的执行过程中,调用recover会返回nil并且不产生其他任何效果。如果当前的goroutine触发了panic,调用recover可以捕获到panic的输入值,并且恢复正常运行。这个特性对于像web服务就非常有用了, 当web服务处理某个请求时,某个方法触发了panic,这时候显然是不应该直接让web服务挂掉的。这种场景下,就可以使用recover来捕获panic并且让服务正常运行下去。

在其他语言里,通常是底层抛出异常,上层逻辑通过try/catch捕获异常。defer/panic/recover配合使用可以实现类似try/catch的功能。

将Recover()写在defer中,在可能发生panic的代码之前执行defer,当程序触发panic后,系统将跳过后面的代码,按照逆序执行已经注册的defer函数,如果defer函数中调用了recover(),recover()会返回捕获到的panic的错误信息。

使用recover需要注意几点:

  • recover需要在defer的方法里面直接调用,不能对recover()包一层方法后再在defer的方法里面调用

  • recover只能捕获同一个协程中的panic,无法捕获其它协程的panic

defer/panic/recover示例

成功捕获实例一

  1. func main() {
  2. defer func() {
  3. if err := recover(); err != nil {
  4. fmt.Printf("recover:%v
  5. ", err)
  6. }
  7. }()
  8. panic("an error occurred")
  9. }

成功捕获实例二

  1. func main() {
  2. defer func() {
  3. if err := recover(); err != nil {
  4. fmt.Printf("recover:%v
  5. ", err)
  6. }
  7. }()
  8. test()
  9. }
  10. func test() {
  11. panic("an error occurred")
  12. }

成功捕获实例三

  1. func main() {
  2. test()
  3. }
  4. func test() {
  5. defer func() {
  6. if err := recover(); err != nil {
  7. fmt.Printf("recover:%v
  8. ", err)
  9. }
  10. }()
  11. panic("an error occurred")
  12. }

不能捕获实例一

  1. func main() {
  2. if err := recover(); err != nil {
  3. fmt.Printf("recover:%v
  4. ", err)
  5. }
  6. panic("an error occurred")
  7. }

不能捕获实例二

  1. func main() {
  2. go func() {
  3. defer func() {
  4. if err := recover(); err != nil {
  5. fmt.Printf("recover:%v
  6. ", err)
  7. }
  8. }()
  9. }()
  10. panic("an error occurred")
  11. }

不能捕获实例三

  1. func main() {
  2. defer func() {
  3. if err := recover(); err != nil {
  4. fmt.Printf("recover:%v
  5. ", err)
  6. }
  7. }()
  8. go test()
  9. for {
  10. select {}
  11. }
  12. }
  13. func test() {
  14. panic("an error occurred")
  15. }

不能捕获实例四

  1. func main() {
  2. defer func() {
  3. recoverFromPanic()
  4. }()
  5. test()
  6. }
  7. func recoverFromPanic() {
  8. if err := recover(); err != nil {
  9. fmt.Printf("recover:%v
  10. ", err)
  11. }
  12. }
  13. func test() {
  14. panic("an error occurred")
  15. }

不能捕获实例五

  1. func main() {
  2. defer func() {
  3. if err := recover(); err != nil {
  4. fmt.Printf("recover:%v
  5. ", err)
  6. }
  7. }()
  8. test()
  9. for {
  10. select {}
  11. }
  12. }
  13. func test() {
  14. go func() {
  15. panic("an error occurred")
  16. }()
  17. }

以上就是Golang中的panic与recover怎么使用的详细内容,更多关于Golang中的panic与recover怎么使用的资料请关注九品源码其它相关文章!