Go语言中的defer,panic,recover 与错误处理
目录
前言
三个关键字
defer语句
panic语句
recover函数
defer、panic、recover组成的错误处理
总结
前言
在其他编程语言中,如Java,宕机往往以异常的形式存在。底层抛出异常,上层逻辑通过try...catch...fanally机制捕获异常并处理,没有被捕获到的严重异常会导致程序崩溃,捕获的异常可以被处理,让代码可以继续执行。
Go语言没有异常系统,其使用panic触发宕机类似于其他语言的抛出异常,而recover的宕机恢复机制就对应try...catch机制。recover()函数用来捕获或者说是拦截panic的,阻止panic继续向上层传递。无论是主动调用panic()函数触发的宕机还是程序在运行过程中由Runtime层抛出的异常,都可以配合defer 和 recover 实现异常捕获和恢复,让代码在发生panic后能够继续执行。
Go语言通过
defer
、panic
和recover
三个关键字构建了一种独特的异常处理机制。它们协同工作,使得Go程序能够优雅地处理运行时错误和异常情况。本文将深入浅出地解析这三个关键字的用法、特点以及常见问题与易错点,并通过代码示例进行演示。
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
三个关键字
defer:在函数中,经常要打开资源(如:文件打开,数据库连接等),为了在函数执行后,及时释放资源,go的设计者提供defer(延时机制)
panic:内置函数,接收一个interface{}类型的值(也就是任何值)作为参数,可接收error类型的变量,输出错误信息,并退出程序。
errors.New("错误说明"):返回一个error类型的值,表示一个错误
recover:内置函数,可以捕获到go中的异常
defer语句
延迟执行
defer
语句用于延迟执行一个函数调用,直到包含该defer
语句的函数返回时才执行。这在资源释放、日志记录等场景中尤为有用:
package mainimport "fmt"func main() {defer fmt.Println("Closing file...")// 执行文件操作...
}// 输出:Closing file...
后进先出(LIFO)
如果有多个defer
语句,它们按后进先出(LIFO)顺序执行:
package mainimport "fmt"func main() {defer fmt.Println("Second deferred call")defer fmt.Println("First deferred call")// 执行其他操作...
}// 输出:
// First deferred call
// Second deferred call
在return语句之后执行
defer
语句的执行时机在函数返回之前,即使它位于return
语句之后:
package mainimport "fmt"func calculate() (result int, err error) {defer func() {if r := recover(); r != nil {err = fmt.Errorf("Recovered from panic: %v", r)}}()// 可能触发panic的计算逻辑...return result, err
}
易错点:滥用defer
导致性能下降。尽管defer
提供了便利,但过多或不必要的使用可能增加函数调用栈的开销。在需要确保资源释放或执行清理操作时合理使用defer
。
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
panic语句
触发运行时错误
panic
语句用于触发一个运行时错误,立即停止当前函数的执行,并开始回溯调用栈,直到遇到recover
或程序终止:
package mainimport "fmt"func mayPanic() {if condition {panic("An error occurred!")}
}func main() {mayPanic()fmt.Println("This line will not be reached.")
}
传递错误信息
panic
可以接受任意类型作为参数,通常传递一个字符串或错误接口实例,以便于错误信息的传递和处理:
package mainimport ("errors""fmt"
)func divide(a, b int) (int, error) {if b == 0 {panic(errors.New("Division by zero"))}return a / b, nil
}func main() {_, err := divide(10, 0)if err != nil {fmt.Println(err) // 输出:Division by zero}
}
易错点:随意使用panic
处理非严重错误。panic
应主要用于处理不可恢复的运行时错误,对于可处理的错误,应通过返回错误值的方式传递给调用者。
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
recover函数
捕获panic
recover
函数只能在defer
语句中调用,用于捕获当前goroutine发生的panic,并返回panic传入的值。如果没有panic发生,recover
返回nil
:
package mainimport "fmt"func mayPanic() {panic("An error occurred!")
}func handlePanic() {defer func() {if r := recover(); r != nil {fmt.Println("Recovered from panic:", r)}}()mayPanic()
}func main() {handlePanic()fmt.Println("Program continues after panic recovery.")
}
易错点:错误地认为recover
可以跨goroutine捕获panic。recover
只能捕获同一goroutine内发生的panic,对于其他goroutine引发的panic无能为力。在并发编程中,应结合sync.Once
、context.Context
等工具实现跨goroutine的错误传播与处理。
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
【##】关于recover()函数的说明
recover()函数被调用后,会返回一个 interface{} 接口类型的返回值,如果返回值等于nil,说明没有触发panic;反之,说明程序发生了panic,就应该采取相应的措施。
示例1:使用recover捕获panic异常,恢复程序的运行。
func funcA() {fmt.Println("func A")
}func funcB() {defer func(){//捕获panic,并恢复程序使其继续运行if err := recover(); err != nil {fmt.Println("recover in funcB")}}()panic("panic in B") //主动抛出异常
}func funcC() {fmt.Println("func C")
}func funcD() {fmt.Println("func D")
}func main() {defer funcA()defer funcC()fmt.Println("this is main")funcB()defer funcD()
}
运行结果:
this is main
recover in funcB
func D
func C
func A
《代码分析》当recover捕获到panic时,不会造成整个进程的崩溃,它会从触发panic的位置退出当前函数,然后继续执行后续代码。
【##】recover()的使用注意事项
- recover 必须搭配defer 语句使用,并且recover()函数必须在延迟函数内被调用执行才能正常工作。
- defer一定要在可能引发panic的语句之前定义。
func funcB() {/*defer func(){//捕获异常,并恢复程序使其继续运行if err := recover(); err != nil {fmt.Println("recover in funcB")}}()*/defer recover() //捕获会失败panic("panic in B") //主动抛出异常
}
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
defer、panic、recover组成的错误处理
go语言不支持 try...catch...finally,取而代之的是 defer、panic、revocer组合
例子一
看以下代码:
package main
import ("fmt"
)
func deferTest2(){var num1 int8 = 10var num2 int8 = 0fmt.Println("函数内部代码")fmt.Println(num1 / num2) // 在这里程序 panic了,因此在该位置退出,且程序不会执行后面的代码
}
func main(){deferTest2()fmt.Println("后面的代码...")
}
运行结果如下:
使用 defer ,revocer 进行错误处理后,任然可以执行后面的代码,因此修改代码如下:
package main
import ("fmt"
)
func deferTest2(){defer func(){ // 函数发生异常后会执行 defer里的代码,并且不会退出程序err := recover()if err != nil {fmt.Println("发生异常:",err)}}()var num1 int8 = 10var num2 int8 = 0fmt.Println("函数内部代码")fmt.Println(num1 / num2)
}
func main(){deferTest2()fmt.Println("后面的代码...")
}
执行结果如下:
例子二,自定义错误,使用panic 输出错误程序后,退出程序
package mainimport ("fmt""errors"
)
func myError(a string) (err error){if a == "a" {return nil}return errors.New("这是一个自定义错误")
}
func doMyError(){err := myError("b")if err != nil { // 如果有错误,则打印错误,并退出程序panic(err) }fmt.Println("继续向后执行...")
}
func main(){doMyError()fmt.Println("后面的代码...")
}
执行结果为:
当 myError 传 a的时候,执行结果如下:
我们再上面代码加入defer语句处理错误后:
import ("fmt""errors"
)
func myError(a string) (err error){if a == "a" {return nil}return errors.New("这是一个自定义错误")
}
func doMyError(){defer func(){err := recover()if err != nil {fmt.Println("出现的error为:",err)}}()err := myError("b")if err != nil {panic(err) // 发生panic 后,执行压入栈的defer语句,且程序不会终止}fmt.Println("继续向后执行...") // 发生panic 后不会执行该语句
}
func main(){doMyError()fmt.Println("后面的代码...")
}
执行结果为:
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
总结
defer:用于延迟执行函数调用,确保在函数返回前执行。它可以用于资源释放、日志记录等场景。多个 defer 语句按照 后进先出 的顺序执行,传参时需要注意变量的求值时机。
panic:用于处理严重错误,立即终止当前函数的执行,并开始回溯调用栈。panic 会触发当前 goroutine 中的所有 defer,但不会影响其他 goroutine。
recover:用于捕获 panic,只能在 defer 延迟函数中使用。通过 recover,你可以在 panic 发生时捕获它,并采取适当的措施来处理错误,从而避免程序崩溃。
defer、panic 、recover和 语句的组合有如下几个特性:
- recover 必须搭配defer 语句使用,并且recover()函数必须在延迟函数内被调用执行才能正常工作。
- defer一定要在可能引发panic的语句之前定义。
- 有panic,没recover,程序宕机。
- 有panic,也有recover,程序不会宕机。执行完对应的defer后,从宕机点退出所在函数返回到主调函数中,继续执行后续代码。
<提示>
在panic触发的defer语句内,可以继续使用panic,进一步将错误外抛直至程序整体崩溃。
如果想在捕获panic时设置当前函数的返回值,可以对返回值使用命名返回值方式直接进行设置。
Go语言提供了两种错误处理的方式,一种是借助 panic 和 recover 的抛出捕获机制,另一种是使用error接口错误类型。在实际的开发工作中,可以根据实际需求选择合适的错误处理方式。
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!
万码优才-数字技术人才找工作上万码优才!招聘求职找工作!万码优才,新一代数字技术人才招聘求职平台。为求职者提供海量高薪职位选择,可通过托管简历让企业主动联系。同时,为企业招聘方提供高效的招人服务,确保优质人才的精准推荐,缩短招聘周期。https://wanmayoucai.com/home?channeltype=drsy&channelid=1310002
相关文章:

Go语言中的defer,panic,recover 与错误处理
目录 前言 三个关键字 defer语句 panic语句 recover函数 defer、panic、recover组成的错误处理 总结 前言 在其他编程语言中,如Java,宕机往往以异常的形式存在。底层抛出异常,上层逻辑通过try...catch...fanally机制捕获异常并处理&am…...
(C语言)力扣 904.水果成篮
写在所有的前面: 本文采用C语言实现代码 目录 写在所有的前面:题目说明题目:力扣 904.水果成篮题目出处题目描述Description输入Input输出Output样例Sample限制Hint 解答说明方案解题思路一般情况特殊情况 代码实现其他解释 题目说明 题目…...

2024 年12月英语六级CET6听力原文(Lecture部分)
2024 年12月英语六级CET6听力原文(Long Conersation和Passage) 1 牛津大学关于普遍道德准则的研究及相关观点与建议 译文 2 食物颜色对味觉体验及大脑预期的影响 译文 3 财务资源对意义与幸福之间关系的影响研究 译文...

CentOS下,离线安装vscode的步骤;
前置条件: 1.CentOS7; 步骤: 1.下载vscode指定版本,例如; 例如 code-1.83.1-1696982959.el7.x86_64.rpm 2.使用下面命令: sudo rpm -ivh code-1.83.1-1696982959.el7.x86_64.rpm 其他: 卸载vscode的命…...
ubuntu停止.netcore正在运行程序的方法
在Ubuntu系统中停止正在运行的.NET Core程序,你可以使用以下几种方法: 使用kill命令: 如果你知道.NET Core程序的进程ID(PID),你可以直接使用kill命令来停止它。首先,使用ps命令配合grep来查找.…...

机器学习基础 衡量模型性能指标
目录 1 前言 编辑1.1 错误率(Error rate)&精度(Accuracy)&误差(Error): 1.2 过拟合(overfitting): 训练误差小,测试误差大 1.3 欠拟合(underfitting):训练误差大,测试误差大 1.4 MSE: 1.5 RMSE: 1.6 MAE: 1.7 R-S…...
《OpenCV计算机视觉》-对图片的各种操作(均值、方框、高斯、中值滤波处理)及形态学处理
文章目录 《OpenCV计算机视觉》-对图片的各种操作(均值、方框、高斯、中值滤波处理)边界填充阈值处理图像平滑处理生成椒盐图片均值滤波处理方框滤波处理高斯滤波处理中值滤波处理 图像形态学腐蚀膨胀开运算闭运算顶帽和黑帽 《OpenCV计算机视觉》-对图片…...

如何让Tplink路由器自身的IP网段 与交换机和电脑的IP网段 保持一致?
问题分析: 正常情况下,我的需求是:电脑又能上网,又需要与路由器处于同一局域网下(串流Pico4 VR眼镜),所以,我是这么连接 交换机、路由器、电脑 的: 此时,登录…...

【JetPack】Navigation知识点总结
Navigation的主要元素: 1、Navigation Graph: 一种新的XML资源文件,包含应用程序所有的页面,以及页面间的关系。 <?xml version"1.0" encoding"utf-8"?> <navigation xmlns:android"http://schemas.a…...

InnoDB引擎的内存结构
InnoDB擅长处理事务,具有自动崩溃恢复的特性 架构图: 由4部分组成: 1.Buffer Pool:缓冲池,缓存表数据和索引数据,减少磁盘I/O操作,提升效率 2.change Buffer:写缓冲区,…...

Y3地图制作1:水果缤纷乐、密室逃脱
文章目录 一、水果缤纷乐1.1 游戏设计1.1.1 项目解析1.1.2 项目优化1.1.3 功能拆分 1.2 场景制作1.2.1 场景需求1.2.2 创建主镜头、绘制草稿,构思文案和情景1.2.3 构建场景地图1.2.4 光源与氛围设置 1.3 游戏初始化1.3.1 物编、UI预设置1.3.2 游戏初始化1.3.2 玩家初…...

ESP32_H2(IDF)学习系列-ADC模数转换(连续转换)
一、简介(节选手册) 1 概述 ESP32-H2 搭载了以下模拟外设: • 一个 12 位逐次逼近型模拟数字转换器 (SAR ADC),用于测量最多来自 5 个管脚上的模拟信号。 • 一个温度传感器,用于测量及监测芯片内部温度。 2 SAR ADC 2…...
如何通过TikTok成功引流到独立站
随着短视频平台的迅猛发展,TikTok已成为全球最受欢迎的社交媒体之一,尤其是在年轻用户群体中更是势不可挡。如果你是一个独立站(如电商网站、博客、个人品牌站等)的运营者,那么如何通过TikTok引流到独立站已经成为一个…...

生成签名文件 .keystore
打开java sdk 到bin目录(D:\JDK\Java\jdk1.8.0_202\bin),打开dos窗口执行以下命令: 命令行输入: 1、生成签名文件:(-alias 别名 validity 有效期 9125 天) keytool -genkeypair -v…...
Mono里运行C#脚本3—mono_jit_init
前面已经介绍了配置参数的读取,这样就可以把一些特殊的配置读取进来,完成了用户配置阶段的参数,接着下来就需要进行大工程的建造了。 为什么这样说呢,因为需要解释并执行C#编译的受托管的代码,相当于就是建立一个C#代码运行的虚拟机,而这个虚拟机还是很复杂的,不但要支…...

龙迅#LT6911C适用于HDMI转MIPI/LVDS产品应用,分辨率高达4K30HZ,内置程序,支持KEY(HDCP)!
1. 描述 LT6911C 是一款高性能 HDMI1.4/DP 转 MIPIDSI/CSI/LVDS 芯片,适用于 VR/智能手机/显示应用。 对于 MIPIDSI/CSI 输出,LT6911C具有可配置的单端口或双端口 MIPIDSI/CSI,具有 1 个高速时钟通道和 1~4 个高速数据通道,运行…...
阿里云虚拟主机ecs镜像如何转移到本地virtualbox上
导出阿里云 ECS 镜像 创建快照:登录阿里云 ECS 控制台,找到对应的 ECS 实例,创建一个快照,等待快照创建完成。创建自定义镜像:基于创建好的快照,创建一个自定义镜像,填写镜像名称和描述等信息。导出镜像:镜像创建完成后,点击 “导出镜像”,并授权访问,将镜像导出到阿…...

虚拟机桥接模式
主机Win10,虚拟机xp 1.虚拟机设置中选择桥接模式 2.在虚拟机菜单:编辑>虚拟机网络编辑,点击“更改设置”,可以看到三个网卡,这三个网卡分别对应不同的网络共享模式。桥接模式须使用VMnet0,如果没看到这个网卡&…...

酷睿i7和i5哪个好?i5和i7的区别介绍
在英特尔酷睿处理器家族中,i7与i5作为面向不同用户群体的主流产品,各自承载着不同的性能定位与使用价值。在面对“酷睿i7和i5哪个好”的问题时,答案并非一概而论,而是取决于具体的应用需求、预算考量以及对性能与效率的期待。本文…...
STM32 高级 谈一下IPV4/默认网关/子网掩码/DNS服务器/MAC
首先可以通过 winr->输入cmd->输入ipconfig 命令可以查看计算机的各种地址 IPV4:是互联网协议第 4 版(Internet Protocol version 4)所使用的地址。它是一个 32 位的二进制数字,通常被分为 4 个 8 位的部分ÿ…...

上门预约行业技术方案全解析:小程序、App还是H5?如何选择?
上门按摩行业这几年确实火,但千万别以为随便买个系统、招几个技师就能赚钱。作为一家深耕10年软件开发的公司,我们做了四五年上门按摩系统,见过太多人头脑一热冲进来,结果血本无归。 如果你真的想做上门按摩,先想清楚这…...
场景题-3
如何实现一个消息队列 拆解分析主流的几种消息队列 1、基本架构 生产者Producer、消费者Consumer、Broker:生产者发送消息,消费者接受消息,Broker是服务端,处理消息的存储、备份、删除和消费关系的维护。 主题和分区ÿ…...

【行驶证识别成表格】批量OCR行驶证识别与Excel自动化处理系统,行驶证扫描件和照片图片识别后保存为Excel表格,基于QT和华为ocr识别的实现教程
在车辆管理、物流运输、保险理赔等领域,经常需要处理大量的行驶证信息。传统的人工录入方式效率低、易出错,而使用 OCR 技术可以自动识别行驶证图片中的文字信息,极大提高数据处理效率。该系统可以应用于以下场景: 保险公司快速…...

每日算法-250605
每日算法 - 20240605 525. 连续数组 题目描述 给定一个二进制数组 nums , 找到含有相同数量的 0 和 1 的最长连续子数组,并返回该子数组的长度。 思路 前缀和 哈希表 解题过程 核心思想是将问题巧妙地转换为寻找和为特定值的子数组问题。 转换问题:我…...
基于 BGE 模型与 Flask 的智能问答系统开发实践
基于 BGE 模型与 Flask 的智能问答系统开发实践 一、前言 在人工智能快速发展的今天,智能问答系统成为了提升信息检索效率和用户体验的重要工具。本文将详细介绍如何利用 BGE(Base General Embedding)模型、Faiss 向量检索库以及 Flask 框架…...
Github 2025-06-04 C开源项目日报 Top7
根据Github Trendings的统计,今日(2025-06-04统计)共有7个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量C项目7C++项目1Assembly项目1jq:轻量灵活的命令行JSON处理器 创建周期:4207 天开发语言:C协议类型:OtherStar数量:27698 个Fork数量:1538 …...
高效图像处理:使用 Pillow 进行格式转换与优化
高效图像处理:使用 Pillow 进行格式转换与优化 1. 背景引入 在图像处理应用中,格式转换、裁剪、压缩等操作是常见需求。Python 的 Pillow 库基于 PIL(Python Imaging Library),提供 轻量、强大 的图像处理能力,广泛用于 Web 开发、数据分析、机器学习 等领域。 本文将…...
Github 2025-06-06 Java开源项目日报Top10
根据Github Trendings的统计,今日(2025-06-06统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Java项目10TypeScript项目1Java实现的算法集合:使用Gitpod.io进行编辑和贡献 创建周期:2883 天开发语言:Java协议类型:MIT LicenseStar数量…...

【推荐算法】NeuralCF:深度学习重构协同过滤的革命性突破
NeuralCF:深度学习重构协同过滤的革命性突破 一、算法背景知识:协同过滤的演进与局限1.1 协同过滤的发展历程1.2 传统矩阵分解的缺陷 二、算法理论/结构:NeuralCF架构设计2.1 基础NeuralCF结构2.2 双塔模型进阶结构2.3 模型实现流程对比 三、…...
Next.js+prisma开发一
1.初始化Next.js项目 #按版本安装 npx create-next-app13.4.5 如果最新版本 执行:npx create-next-applatest2. 安装Prima和客户端 npm install prisma --save-dev npm install prisma/client3.初始化Prisma,以SQLit举例 # 初始化 Prisma 并配置 SQLi…...