网站首页 > 技术文章 正文
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的功能,下篇文章将从源码角度来做讲解。
猜你喜欢
- 2024-12-17 C语言实现推箱子游戏!(超简单详细)代码思路+源码分享
- 2024-12-17 学习笔记之C#基础——数组和集合 c#中数组用法
- 2024-12-17 新函数REDUCE来了!Excel中的最强辅助,太强大了
- 2024-12-17 一篇文章学会golang语法,golang简明教程快速入门
- 2024-12-17 深入理解 Golang 中的值类型和引用类型
- 2024-12-17 SpringBoot系列之数据库初始化-datasource配置方式
- 2024-12-17 C++ 创建数组和使用数组学习笔记 c++如何建立数组
- 2024-12-17 Java Map 中那些巧妙的设计 javamap的用法
- 2024-12-17 大数据开发基础之一维数组的定义、初始化及与二维数组的区别
- 2024-12-17 go语言结构体与初始化 go 结构体初始化
- 1509℃桌面软件开发新体验!用 Blazor Hybrid 打造简洁高效的视频处理工具
- 527℃Dify工具使用全场景:dify-sandbox沙盒的原理(源码篇·第2期)
- 492℃MySQL service启动脚本浅析(r12笔记第59天)
- 472℃服务器异常重启,导致mysql启动失败,问题解决过程记录
- 469℃启用MySQL查询缓存(mysql8.0查询缓存)
- 450℃「赵强老师」MySQL的闪回(赵强iso是哪个大学毕业的)
- 429℃mysql服务怎么启动和关闭?(mysql服务怎么启动和关闭)
- 426℃MySQL server PID file could not be found!失败
- 最近发表
- 标签列表
-
- c++中::是什么意思 (83)
- 标签用于 (65)
- 主键只能有一个吗 (66)
- c#console.writeline不显示 (75)
- pythoncase语句 (81)
- es6includes (73)
- windowsscripthost (67)
- apt-getinstall-y (86)
- node_modules怎么生成 (76)
- chromepost (65)
- c++int转char (75)
- static函数和普通函数 (76)
- el-date-picker开始日期早于结束日期 (70)
- js判断是否是json字符串 (67)
- checkout-b (67)
- localstorage.removeitem (74)
- vector线程安全吗 (70)
- & (66)
- java (73)
- js数组插入 (83)
- linux删除一个文件夹 (65)
- mac安装java (72)
- eacces (67)
- 查看mysql是否启动 (70)
- 无效的列索引 (74)