一文掌握Golang中Panic与Recover的作用和使用方法

开发 前端
本文介绍了Panic和Recover的作用及使用方法,以及Defer/Panic/Recover配合使用实现类似Try/Catch的功能。

panic

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

package main   
  
func main() {  
    // 未处理的自定义异常  
    customException := "an error occurred"   
    panic(customException)  
}

会输出如下信息:

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示例:

package main

import "fmt"

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

会输出如下信息:

# 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示例

成功捕获实例一

func main() {
	defer func() {
		if err := recover(); err != nil {
			fmt.Printf("recover:%v\n", err)
		}
	}()
	panic("an error occurred")
}

成功捕获实例二

func main() {
	defer func() {
		if err := recover(); err != nil {
			fmt.Printf("recover:%v\n", err)
		}
	}()
	test()
}

func test() {
	panic("an error occurred")
}

成功捕获实例三

func main() {
	test()
}

func test() {
	defer func() {
		if err := recover(); err != nil {
			fmt.Printf("recover:%v\n", err)
		}
	}()
	panic("an error occurred")
}

不能捕获实例一

func main() {
	if err := recover(); err != nil {
		fmt.Printf("recover:%v\n", err)
	}
	panic("an error occurred")
}

不能捕获实例二

func main() {
	go func() {
		defer func() {
			if err := recover(); err != nil {
				fmt.Printf("recover:%v\n", err)
			}
		}()
	}()
	panic("an error occurred")
}

不能捕获实例三

func main() {
	defer func() {
		if err := recover(); err != nil {
			fmt.Printf("recover:%v\n", err)
		}
	}()

	go test()

	for {
		select {}
	}
}

func test() {
	panic("an error occurred")
}

不能捕获实例四

func main() {
	defer func() {
		recoverFromPanic()
	}()

	test()
}

func recoverFromPanic() {
	if err := recover(); err != nil {
		fmt.Printf("recover:%v\n", err)
	}
}

func test() {
	panic("an error occurred")
}

不能捕获实例五

func main() {
	defer func() {
		if err := recover(); err != nil {
			fmt.Printf("recover:%v\n", err)
		}
	}()

	test()

	for {
		select {}
	}
}

func test() {
	go func() {
		panic("an error occurred")
	}()
}

小结

本文介绍了panic和recover的作用及使用方法,以及defer/panic/recover配合使用实现类似try/catch的功能,下篇文章将从源码角度来做讲解。

责任编辑:姜华 来源: 今日头条
相关推荐

2023-03-03 13:43:00

Java字节流

2023-10-09 07:14:42

panicGo语言

2023-08-01 09:27:44

Golang模糊测试

2023-07-04 08:56:07

指针类型Golang

2022-06-09 08:17:30

Python__new__

2024-02-23 19:11:13

C++编程开发

2010-07-05 09:38:47

LinuxNFS

2022-10-21 17:24:34

契约测试定位

2023-11-30 07:15:36

GolangRecover

2009-08-21 18:00:38

ASP.NET mac

2022-12-20 07:39:46

2023-12-21 17:11:21

Containerd管理工具命令行

2023-05-11 08:00:44

Golangsync.Pool

2021-06-06 13:06:34

JVM内存分布

2021-09-09 17:05:36

C++智能指针语言

2023-09-16 19:38:17

Python私有属性私有方法

2023-06-28 08:34:02

Bind()函数JavaScript

2009-11-17 17:38:37

PHP Session

2021-10-28 10:26:35

Javascript 高阶函数前端

2020-10-09 07:56:52

Linux
点赞
收藏

51CTO技术栈公众号