当前位置: 首页 > news >正文

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语言通过deferpanicrecover三个关键字构建了一种独特的异常处理机制。它们协同工作,使得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.Oncecontext.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接口错误类型。在实际的开发工作中,可以根据实际需求选择合适的错误处理方式。

万码优才-数字技术人才找工作上万码优才!招聘求职找工作!

万码优才-数字技术人才找工作上万码优才!招聘求职找工作!万码优才,新一代数字技术人才招聘求职平台。为求职者提供海量高薪职位选择,可通过托管简历让企业主动联系。同时,为企业招聘方提供高效的招人服务,确保优质人才的精准推荐,缩短招聘周期。icon-default.png?t=O83Ahttps://wanmayoucai.com/home?channeltype=drsy&channelid=1310002

相关文章:

Go语言中的defer,panic,recover 与错误处理

目录 前言 三个关键字 defer语句 panic语句 recover函数 defer、panic、recover组成的错误处理 总结 前言 在其他编程语言中&#xff0c;如Java&#xff0c;宕机往往以异常的形式存在。底层抛出异常&#xff0c;上层逻辑通过try...catch...fanally机制捕获异常并处理&am…...

(C语言)力扣 904.水果成篮

写在所有的前面&#xff1a; 本文采用C语言实现代码 目录 写在所有的前面&#xff1a;题目说明题目&#xff1a;力扣 904.水果成篮题目出处题目描述Description输入Input输出Output样例Sample限制Hint 解答说明方案解题思路一般情况特殊情况 代码实现其他解释 题目说明 题目…...

2024 年12月英语六级CET6听力原文(Lecture部分)

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

CentOS下,离线安装vscode的步骤;

前置条件&#xff1a; 1.CentOS7; 步骤&#xff1a; 1.下载vscode指定版本&#xff0c;例如&#xff1b; 例如 code-1.83.1-1696982959.el7.x86_64.rpm 2.使用下面命令&#xff1a; sudo rpm -ivh code-1.83.1-1696982959.el7.x86_64.rpm 其他&#xff1a; 卸载vscode的命…...

ubuntu停止.netcore正在运行程序的方法

在Ubuntu系统中停止正在运行的.NET Core程序&#xff0c;你可以使用以下几种方法&#xff1a; 使用kill命令&#xff1a; 如果你知道.NET Core程序的进程ID&#xff08;PID&#xff09;&#xff0c;你可以直接使用kill命令来停止它。首先&#xff0c;使用ps命令配合grep来查找.…...

机器学习基础 衡量模型性能指标

目录 1 前言 ​编辑1.1 错误率(Error rate)&精度(Accuracy)&误差(Error)&#xff1a; 1.2 过拟合(overfitting): 训练误差小&#xff0c;测试误差大 1.3 欠拟合(underfitting)&#xff1a;训练误差大&#xff0c;测试误差大 1.4 MSE: 1.5 RMSE: 1.6 MAE: 1.7 R-S…...

《OpenCV计算机视觉》-对图片的各种操作(均值、方框、高斯、中值滤波处理)及形态学处理

文章目录 《OpenCV计算机视觉》-对图片的各种操作&#xff08;均值、方框、高斯、中值滤波处理&#xff09;边界填充阈值处理图像平滑处理生成椒盐图片均值滤波处理方框滤波处理高斯滤波处理中值滤波处理 图像形态学腐蚀膨胀开运算闭运算顶帽和黑帽 《OpenCV计算机视觉》-对图片…...

如何让Tplink路由器自身的IP网段 与交换机和电脑的IP网段 保持一致?

问题分析&#xff1a; 正常情况下&#xff0c;我的需求是&#xff1a;电脑又能上网&#xff0c;又需要与路由器处于同一局域网下&#xff08;串流Pico4 VR眼镜&#xff09;&#xff0c;所以&#xff0c;我是这么连接 交换机、路由器、电脑 的&#xff1a; 此时&#xff0c;登录…...

【JetPack】Navigation知识点总结

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

InnoDB引擎的内存结构

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

Y3地图制作1:水果缤纷乐、密室逃脱

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

ESP32_H2(IDF)学习系列-ADC模数转换(连续转换)

一、简介&#xff08;节选手册&#xff09; 1 概述 ESP32-H2 搭载了以下模拟外设&#xff1a; • 一个 12 位逐次逼近型模拟数字转换器 (SAR ADC)&#xff0c;用于测量最多来自 5 个管脚上的模拟信号。 • 一个温度传感器&#xff0c;用于测量及监测芯片内部温度。 2 SAR ADC 2…...

如何通过TikTok成功引流到独立站

随着短视频平台的迅猛发展&#xff0c;TikTok已成为全球最受欢迎的社交媒体之一&#xff0c;尤其是在年轻用户群体中更是势不可挡。如果你是一个独立站&#xff08;如电商网站、博客、个人品牌站等&#xff09;的运营者&#xff0c;那么如何通过TikTok引流到独立站已经成为一个…...

生成签名文件 .keystore

打开java sdk 到bin目录&#xff08;D:\JDK\Java\jdk1.8.0_202\bin&#xff09;&#xff0c;打开dos窗口执行以下命令&#xff1a; 命令行输入&#xff1a; 1、生成签名文件&#xff1a;&#xff08;-alias 别名 validity 有效期 9125 天&#xff09; 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 芯片&#xff0c;适用于 VR/智能手机/显示应用。 对于 MIPIDSI/CSI 输出&#xff0c;LT6911C具有可配置的单端口或双端口 MIPIDSI/CSI&#xff0c;具有 1 个高速时钟通道和 1~4 个高速数据通道&#xff0c;运行…...

阿里云虚拟主机ecs镜像如何转移到本地virtualbox上

导出阿里云 ECS 镜像 创建快照:登录阿里云 ECS 控制台,找到对应的 ECS 实例,创建一个快照,等待快照创建完成。创建自定义镜像:基于创建好的快照,创建一个自定义镜像,填写镜像名称和描述等信息。导出镜像:镜像创建完成后,点击 “导出镜像”,并授权访问,将镜像导出到阿…...

虚拟机桥接模式

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

酷睿i7和i5哪个好?i5和i7的区别介绍

在英特尔酷睿处理器家族中&#xff0c;i7与i5作为面向不同用户群体的主流产品&#xff0c;各自承载着不同的性能定位与使用价值。在面对“酷睿i7和i5哪个好”的问题时&#xff0c;答案并非一概而论&#xff0c;而是取决于具体的应用需求、预算考量以及对性能与效率的期待。本文…...

STM32 高级 谈一下IPV4/默认网关/子网掩码/DNS服务器/MAC

首先可以通过 winr->输入cmd->输入ipconfig 命令可以查看计算机的各种地址 IPV4&#xff1a;是互联网协议第 4 版&#xff08;Internet Protocol version 4&#xff09;所使用的地址。它是一个 32 位的二进制数字&#xff0c;通常被分为 4 个 8 位的部分&#xff…...

浅谈 React Hooks

React Hooks 是 React 16.8 引入的一组 API&#xff0c;用于在函数组件中使用 state 和其他 React 特性&#xff08;例如生命周期方法、context 等&#xff09;。Hooks 通过简洁的函数接口&#xff0c;解决了状态与 UI 的高度解耦&#xff0c;通过函数式编程范式实现更灵活 Rea…...

java_网络服务相关_gateway_nacos_feign区别联系

1. spring-cloud-starter-gateway 作用&#xff1a;作为微服务架构的网关&#xff0c;统一入口&#xff0c;处理所有外部请求。 核心能力&#xff1a; 路由转发&#xff08;基于路径、服务名等&#xff09;过滤器&#xff08;鉴权、限流、日志、Header 处理&#xff09;支持负…...

高等数学(下)题型笔记(八)空间解析几何与向量代数

目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...

【2025年】解决Burpsuite抓不到https包的问题

环境&#xff1a;windows11 burpsuite:2025.5 在抓取https网站时&#xff0c;burpsuite抓取不到https数据包&#xff0c;只显示&#xff1a; 解决该问题只需如下三个步骤&#xff1a; 1、浏览器中访问 http://burp 2、下载 CA certificate 证书 3、在设置--隐私与安全--…...

Spring Boot面试题精选汇总

&#x1f91f;致敬读者 &#x1f7e9;感谢阅读&#x1f7e6;笑口常开&#x1f7ea;生日快乐⬛早点睡觉 &#x1f4d8;博主相关 &#x1f7e7;博主信息&#x1f7e8;博客首页&#x1f7eb;专栏推荐&#x1f7e5;活动信息 文章目录 Spring Boot面试题精选汇总⚙️ **一、核心概…...

C# 类和继承(抽象类)

抽象类 抽象类是指设计为被继承的类。抽象类只能被用作其他类的基类。 不能创建抽象类的实例。抽象类使用abstract修饰符声明。 抽象类可以包含抽象成员或普通的非抽象成员。抽象类的成员可以是抽象成员和普通带 实现的成员的任意组合。抽象类自己可以派生自另一个抽象类。例…...

Matlab | matlab常用命令总结

常用命令 一、 基础操作与环境二、 矩阵与数组操作(核心)三、 绘图与可视化四、 编程与控制流五、 符号计算 (Symbolic Math Toolbox)六、 文件与数据 I/O七、 常用函数类别重要提示这是一份 MATLAB 常用命令和功能的总结,涵盖了基础操作、矩阵运算、绘图、编程和文件处理等…...

Ascend NPU上适配Step-Audio模型

1 概述 1.1 简述 Step-Audio 是业界首个集语音理解与生成控制一体化的产品级开源实时语音对话系统&#xff0c;支持多语言对话&#xff08;如 中文&#xff0c;英文&#xff0c;日语&#xff09;&#xff0c;语音情感&#xff08;如 开心&#xff0c;悲伤&#xff09;&#x…...

C# SqlSugar:依赖注入与仓储模式实践

C# SqlSugar&#xff1a;依赖注入与仓储模式实践 在 C# 的应用开发中&#xff0c;数据库操作是必不可少的环节。为了让数据访问层更加简洁、高效且易于维护&#xff0c;许多开发者会选择成熟的 ORM&#xff08;对象关系映射&#xff09;框架&#xff0c;SqlSugar 就是其中备受…...

CSS设置元素的宽度根据其内容自动调整

width: fit-content 是 CSS 中的一个属性值&#xff0c;用于设置元素的宽度根据其内容自动调整&#xff0c;确保宽度刚好容纳内容而不会超出。 效果对比 默认情况&#xff08;width: auto&#xff09;&#xff1a; 块级元素&#xff08;如 <div>&#xff09;会占满父容器…...