17 go语言(golang) - 错误处理
错误处理
错误处理是编程中用于识别、响应和恢复程序运行时出现的错误和异常情况的过程。其目的是确保程序的鲁棒性(一个系统、模型或函数在面对错误输入、工作压力、意外情况或故意攻击时仍能保持稳定性和可靠性的能力),即使在出现错误的情况下也能正常运行或优雅地终止。
而Go语言中的错误处理是一种显式的、基于值的机制,与许多其他编程语言使用异常(exception)不同。Go通过返回值来处理错误,这使得代码更加清晰和可预测。
基本概念
error接口
-
Go内置了一个名为
error的接口,用于表示错误。源码:// The error built-in interface type is the conventional interface for // representing an error condition, with the nil value representing no error. // 内置的error接口类型是表示错误条件的常规接口,其nil值代表没有错误。 type error interface {Error() string } -
任何实现了这个接口的方法都可以作为一个错误对象。
自定义error
-
使用
errors.New或fmt.Errorf()创建简单的文本描述性错误。func Test1(t *testing.T) {err1 := errors.New("自定义错误1")fmt.Println(err1.Error())err2 := fmt.Errorf("自定义错误2")fmt.Println(err2) }输出
自定义错误1 自定义错误2 -
如果需要更复杂或结构化的信息,可以定义自己的类型并实现
Error()方法。type MyError struct {code intmsg string }func (m *MyError) Error() string {return fmt.Sprintf("code:%d,msg:%s", m.code, m.msg) }func Test2(t *testing.T) {myError := MyError{404,"自定义错误!",}fmt.Println(myError.Error())fmt.Println(myError)}输出
code:404,msg:自定义错误! {404 自定义错误!}
函数返回值
-
在Go中,函数通常会返回两个值:结果和一个表示可能发生的错误。
func testDivide(a float64, b float64) (float64, error) {if b == 0 {return 0, fmt.Errorf("分母不能为0")} else {return a / b, nil}}
检查和处理错误
- 调用函数时,需要检查第二个返回值是否为非nil,以判断是否发生了错误。
func Test3(t *testing.T) {var a float64 = 1var b float64divide, err := testDivide(a, b)if err != nil {fmt.Println(divide, err.Error())} else {fmt.Printf("结果为:%.2f\n", divide)}}
输出
0 分母不能为0
错误包装与解包
-
Go1.13引入了对error wrapping的支持,通过使用
fmt.Errorf()和%w格式化动词,可以将上下文信息附加到原始错误上。 -
使用
errors.Unwrap()可以提取被包装过的一层的错误。 -
errors.Is函数用于检查一个错误是否与目标错误相同,或者是否是目标错误的一部分。
func Test4(t *testing.T) {err1 := errors.New("第一层错误")err2 := fmt.Errorf("第二层:%w", err1)fmt.Println(err1)fmt.Println(err2)// errors.Is函数用于检查一个错误是否与目标错误相同,或者是否是目标错误的一部分。// 注意参数顺序,检查第一个错误是不是第二个错误的子错误,或是否同一个错误bool1 := errors.Is(err1, err2) // 返回falsefmt.Println(bool1)bool2 := errors.Is(err2, err1)fmt.Println(bool2)err3 := errors.Unwrap(err2)fmt.Println(err3)
}
输出:
第一层错误
第二层:第一层错误
false
true
第一层错误
错误类型断言与比较
- 使用类型断言来判断特定类型的自定义异常,并访问其内部字段。
- (推荐)
errors.As函数用于检查一个错误链中是否存在某个特定类型的错误,并且可以将该错误赋值给目标变量。它不仅适用于简单的类型断言,还能处理被包装过的复杂错误结构。
func Test5(t *testing.T) {myError := MyError{code: 400,msg: "自定义错误",}var myErr error = &myError// 假设我们只知道这是个错误,但不知道具体的类型时,可以这样处理if err, ok := myErr.(*MyError); ok {fmt.Println(err.code, err.msg)}// 但是官方还是推荐我们使用as来判断具体的错误类型var targetErr *MyErrorif errors.As(myErr, &targetErr) {fmt.Println(targetErr.code, targetErr.msg)}}
输出
400 自定义错误
400 自定义错误
断言只能判断是否那个类型,而error.As能查找是否该类型或子类型
func Test6(t *testing.T) {myError := MyError{code: 400,msg: "自定义错误",}var myErr error = &myError// 再封装一层var myErrNew = fmt.Errorf("封装多一层:%w", myErr)// 假设我们只知道这是个错误,但不知道具体的类型时,可以这样处理if err, ok := myErrNew.(*MyError); ok {fmt.Println("1:", err.code, err.msg)}// 但是官方还是推荐我们使用as来判断具体的错误类型var targetErr *MyErrorif errors.As(myErrNew, &targetErr) {fmt.Println("2:", targetErr.code, targetErr.msg)}}
输出
2: 400 自定义错误
panic和recover
panic
panic是Go语言中的一种内建机制,用于表示程序遇到了无法恢复的严重错误,导致程序的正常执行流程被中断。它类似于其他编程语言中的异常,但在Go中主要用于不可恢复的错误情况。虽然不推荐在正常业务逻辑中使用 panic,但在一些无法恢复或者必须立即停止程序执行情况下可以考虑,比如数组越界等。
关键特点
-
立即停止控制流:
- 当
panic被调用时,程序会立即停止当前函数的执行,并开始运行任何已注册的defer函数。
- 当
-
向上传播:
panic会沿着调用栈向上传播,逐层退出每个调用函数,并在每个退出点运行相应的defer语句。
-
终止程序:
- 如果没有捕获到并处理这个
panic(即没有使用recover),最终会导致整个程序崩溃并打印出堆栈跟踪信息。
- 如果没有捕获到并处理这个
使用场景
- 通常用于那些不应该发生、且无法继续安全运行下去的问题,例如数组越界、空指针解引用等。
- 不建议在普通业务逻辑中使用,而是保留给真正需要立即停止执行的问题。
func Test1(t *testing.T) {defer fmt.Println("执行defer")fmt.Println("程序正常执行")panic("遇到错误!")fmt.Println("程序继续执行")}
输出
=== RUN Test1
程序正常执行
执行defer
--- FAIL: Test1 (0.00s)
panic: 遇到错误! [recovered]panic: 遇到错误!goroutine 6 [running]:
testing.tRunner.func1.2({0xa25ade0, 0xa284890}).........
recover
recover是Go语言中用于处理panic的一种机制。它允许程序在发生panic后恢复执行,从而避免程序崩溃。通常与defer一起使用,以确保即使在函数发生恐慌时也能执行一些清理操作。
基本用法
- 捕获panic(恐慌):当一个函数调用了
panic()时,程序会立即停止当前函数的执行,并开始沿着调用栈向上传播,直到遇到一个能够处理该panic的地方。如果在传播过程中遇到了一个包含recover()的延迟(deferred)函数调用,那么可以通过recover()捕获这个恐慌。 - 返回值:如果成功捕获到一个正在传播中的恐慌,
recover()会返回传递给该panic()调用的值;如果没有处于panic状态,则返回 nil。
使用场景
- 防止程序崩溃:通过捕获和处理不可预见或致命错误,使得程序能够继续运行或进行适当降级。
- 清理资源:确保无论是否出现错误,都能正确释放资源(如文件句柄、网络连接等)。
func Test2(t *testing.T) {defer func() {a := recover()if a != nil {fmt.Println("recover捕获到panic:", a)} else {fmt.Println("recover没有捕获到panic")}}()fmt.Println("程序正常执行")panic("遇到错误!")fmt.Println("程序继续执行")}
输出:
程序正常执行
recover捕获到panic: 遇到错误!
和throw相似
学过其他语言特别是java的同学应该发现,这与Java中的throw关键字非常相似。
尽管panic和throw在概念上相似,但它们在各自的语言中有着不同的使用习惯和语义。在Go中,panic通常用于处理真正的异常情况,比如程序的内部错误,而错误处理则通过返回error类型来完成。在Java中,throw关键字则用于更广泛的场景,包括业务逻辑中预期的异常情况。
-
主动抛出异常:在Go中使用
panic和在Java中使用throw都可以主动抛出一个异常或错误条件。 -
改变程序控制流:两者都会导致程序的正常控制流被改变。在Go中,
panic会立即停止当前函数的执行,并开始执行defer语句,然后向上返回,直到被recover捕获或者程序终止。而在Java中,如果没有在该方法内部捕获,则需要由上层调用者负责处理,否则可能会导致程序崩溃。 -
异常传播:在两个语言中,如果抛出的异常没有被立即捕获,它会继续向上传播到调用栈中的更高层,直到被捕获或者导致程序终止。
-
用于不可恢复的错误:
panic和throw都用于表示那些不应该被忽略的严重错误,它们通常用于指示程序遇到了不可恢复的状态。 -
停止执行后续代码:在
panic或throw之后的代码将不会被执行,直到异常被处理。 -
堆栈跟踪:当
panic发生时,Go会打印出堆栈跟踪信息,类似于Java中throw异常时的行为,这有助于开发者定位错误发生的位置。
相关文章:
17 go语言(golang) - 错误处理
错误处理 错误处理是编程中用于识别、响应和恢复程序运行时出现的错误和异常情况的过程。其目的是确保程序的鲁棒性(一个系统、模型或函数在面对错误输入、工作压力、意外情况或故意攻击时仍能保持稳定性和可靠性的能力),即使在出现错误的情…...
PG 库停库超时异常案例
文章目录 现象官方文档停库底层流程:恢复脚本优化思路总结 现象 停库超时 <2024-11-29 12:50:43.022 UTC 87472 192.167.60.1(54862) PostgreSQL JDBC Driver postgres stk>FATAL: terminating connection due to administrator command <2024-11-29 12:50:43.022 …...
redis下载、基础数据类型、操作讲解说明,持久化、springboot整合等
1 Redis是什么 官网:https://redis.io 开发者:Antirez Redis诞生于2009年全称是Remote Dictionary Server 远程词典服务器,是一个基于内存的键值型NoSQL数据库。 Redis是一个开源的、高性能的键值对存储系统,它支持多种数据结构&…...
[代码随想录06]哈希表的使用,有效字母异位词,两数组交集,快乐数,两数之和
前言 哈希表是什么?一句话带你理解,简单来说我们对于杂乱的数据,怎么快速找到数据,如何做呢?一般的做法就是遍历复杂度为o(N)去找寻一个数据,但是吧,我们这样思考的话,还是花了大量时…...
【CSS】一篇掌握CSS
不是因为有了希望才去坚持,而是坚持了才有了希望 目录 一.导入方式 1.行内样式 2.内部样式 3.外部样式(常用) 二.选择器 1.基本选择器(常用) 1.1标签选择器 1.2类选择器 1.3id选择器 2.层次选择器 2.1后代选择器 2.2子选择器 2.3相邻兄弟选择器 2.4通用兄弟选择器…...
分层图最短路
常见情形: 对于边有k次操作的题。。 整体思想: 分层图最短路可以视作是dijkstra的一个扩展,通常用于处理N小于10000,或者是k不大的情形。整体有点类似于拆点。将一个点拆成k个点处理。层与层之间互不影响。 好了我就说这么多&…...
vue3 基本使用
Vue 3 提供了多种方式来构建用户界面,包括选项式 API 和 Composition API。下面我将详细介绍 Vue 3 的基本使用和语法,主要集中在选项式 API 上,因为这对于初学者来说更容易上手。 1. 创建 Vue 项目 如果你还没有一个 Vue 项目,…...
【maven-4】IDEA 配置本地 Maven 及如何使用 Maven 创建 Java 工程
IntelliJ IDEA(以下简称 IDEA)是一款功能强大的集成开发环境,广泛应用于 Java 开发。下面将详细介绍如何在 IDEA 中配置本地 Maven,并创建一个 Maven Java 工程,快速上手并高效使用 Maven 进行 Java 开发。 1. Maven …...
种花问题算法
假设有一个很长的花坛,一部分地块种植了花,另一部分却没有。可是,花不能种植在相邻的地块上,它们会争夺水源,两者都会死去。 给你一个整数数组 flowerbed 表示花坛,由若干 0 和 1 组成,其中 0 …...
对于大规模的淘宝API接口数据,有什么高效的处理方法?
1.数据分批处理 原理:当处理大规模数据时,一次性将所有数据加载到内存中可能会导致内存溢出。将数据分成较小的批次进行处理可以有效避免这个问题。示例代码:假设通过淘宝 API 获取到了一个包含大量商品详情的 JSON 数据列表,每个…...
openharmony 使用uvc库获取摄像头数据使用nativewindow显示
界面代码: XComponent({ id: xcomponentId, type: texture, libraryname: entry }).width(800).height(500) Natvie代码: 1、头文件 //NativeWindow #include <ace/xcomponent/native_interface_xcomponent.h> #include <cstdint> #incl…...
SQL Server 实战 - 多种连接
目录 背景 一、多种连接 1. 复合连接条件 2. 跨数据库连接 3. 隐连接 4. 自连接 5. 多表外连接 6. UNION ALL 二、一个对比例子 背景 本专栏文章以 SAP 实施顾问在实施项目中需要掌握的 sql 语句为偏向进行选题: 用例:SAP B1 的数据库工具&am…...
【手术显微镜】市场高度集中,由于高端手术显微镜的制造技术主要掌握于欧美企业
摘要 HengCe (恒策咨询)是全球知名的大型咨询机构,长期专注于各行业细分市场的调研。行业层面,重点关注可能存在“卡脖子”的高科技细分领域。企业层面,重点关注在国际和国内市场在规模和技术等层面具有代表性的企业,…...
IDEA 2024 配置Maven
Step 1:确定下载Apache Maven版本 在IDEA 2024中,随便新建一个Maven项目; 在File下拉菜单栏中,找到Setings; 在Build,Execution,Deployment中找到Maven 确定下载的Apache Maven版本应略低于或等于IDEA绑…...
Admin.NET框架使用宝塔面板部署步骤
文章目录 Admin.NET框架使用宝塔面板部署步骤🎁框架介绍部署步骤1.Centos7 部署宝塔面板2.部署Admin.NET后端3.部署前端Web4.访问前端页面 Admin.NET框架使用宝塔面板部署步骤 🎁框架介绍 Admin.NET 是基于 .NET6 (Furion/SqlSugar) 实现的通用权限开发…...
Flutter中的Future和Stream
在 Flutter 中,Future 和 Stream 都是用于处理异步操作的类,它们都基于 Dart 的异步编程模型,但是它们的使用场景和工作方式有所不同。以下是它们的区别以及各自适用的场景。 目录 一、Future1、基本使用2、异常处理1. catchError2. onError…...
107.【C语言】数据结构之二叉树求总节点和第K层节点的个数
目录 1.求二叉树总的节点的个数 1.容易想到的方法 代码 缺陷 思考:能否在TreeSize函数内定义静态变量解决size的问题呢? 其他写法 运行结果 2.最好的方法:分而治之 代码 运行结果 2.求二叉树第K层节点的个数 错误代码 运行结果 修正 运行结果 其他写法 1.求二…...
spring boot支持那些开发工具?
Spring Boot 支持多种开发工具,以帮助开发者更高效地进行应用开发。以下是小编给大家分享几种常用的开发工具及其特点: IntelliJ IDEA: IntelliJ IDEA 是一款非常流行的 Java IDE,它提供了对 Spring Boot 的全面支持,…...
Go-MediatR:Go语言中的中介者模式
在Go语言中,确实存在一个与C#中的MediatR类似的组件包,名为Go-MediatR。 Go-MediatR是一个受.NET中MediatR库启发的Go语言实现,它专注于通过中介者模式简化命令查询责任分离(CQRS)模式的处理和在事件驱动架构中的应用…...
5.11【机器学习】
先是对图像进行划分 划分完后, 顺序读取文件夹,在文件夹里顺序读取图片, 卷积层又称为滤波器,通道是说滤波器的个数,黑白通道数为1,RGB通道个数为3 在输入层,对于输入层而言,滤波…...
ChromePass终极指南:浏览器密码提取与安全管理完全攻略
ChromePass终极指南:浏览器密码提取与安全管理完全攻略 【免费下载链接】chromepass Get all passwords stored by Chrome on WINDOWS. 项目地址: https://gitcode.com/gh_mirrors/chr/chromepass 副标题:从密码危机到数据掌控:3步实现…...
无需编程!DouyinLiveWebFetcher让运营人员轻松实现抖音直播弹幕实时采集
无需编程!DouyinLiveWebFetcher让运营人员轻松实现抖音直播弹幕实时采集 【免费下载链接】DouyinLiveWebFetcher 抖音直播间网页版的弹幕数据抓取(2024最新版本) 项目地址: https://gitcode.com/gh_mirrors/do/DouyinLiveWebFetcher 如…...
苹果全球推出关键MDM工具和企业服务
随着苹果在企业市场份额的稳步增长,该公司终于在美国以外地区推出了其面向中小型企业(SMB)的实用服务集合Apple Business Essentials,但这次它不再叫Apple Business Essentials,而且其中大部分服务都将免费提供。Apple…...
SparkFun ICM-20948 Arduino库:DMP硬件协处理器深度实践指南
1. 项目概述SparkFun ICM-20948 Arduino Library 是面向 TDK InvenSense ICM-20948 九轴惯性测量单元(9DoF IMU)的官方 Arduino 封装库,专为 SparkFun 9DoF IMU Breakout - ICM-20948(Qwiic 接口版本,型号 SEN-15335&a…...
基于粒子群优化算法的永磁同步电机PMSM参数辨识:‘粒子群迭代‘至‘再次循环或结束
基于粒子群优化算法的永磁同步电机PMSM参数辨识 关键词:永磁同步电机 粒子群优化算法 参数辨识 ① 粒子群迭代 ②更新速度并对速度进行边界处理 ③更新位置并对位置进行边界处理 ④进行自适应变异 ⑤进行约束条件判断并计算新种群各个个体位置的适应度 ⑥新适应度与…...
AT25SF041 SPI Flash驱动设计与嵌入式可靠性实践
1. AT25SF041 SPI Flash 存储器驱动深度解析AT25SF041 是由 Adesto(现为 Dialog Semiconductor)推出的 4 Mbit(512 KB)串行 NOR Flash 存储器,采用标准四线 SPI 接口(CLK、CS#、DI、DO)…...
SRAM vs ReRAM vs Flash:一张表看懂不同存内计算芯片的优缺点与选型指南
SRAM vs ReRAM vs Flash:存内计算芯片技术选型全景指南 在AI算力需求爆炸式增长的今天,传统冯诺依曼架构的"内存墙"瓶颈日益凸显。存内计算技术通过将计算单元嵌入存储阵列,彻底打破了数据搬运的能耗桎梏。根据最新行业报告&#x…...
通义千问3-Reranker-0.6B效果惊艳:数学证明步骤间逻辑连贯性重排序
通义千问3-Reranker-0.6B效果惊艳:数学证明步骤间逻辑连贯性重排序 1. 模型介绍与核心能力 通义千问3-Reranker-0.6B是Qwen3 Embedding模型系列的最新成员,专门针对文本重排序任务进行了深度优化。这个6亿参数的模型虽然体积小巧,但在数学证…...
突破运营商限制:中兴光猫配置文件解密工具完全指南
突破运营商限制:中兴光猫配置文件解密工具完全指南 【免费下载链接】ZET-Optical-Network-Terminal-Decoder 项目地址: https://gitcode.com/gh_mirrors/ze/ZET-Optical-Network-Terminal-Decoder 一、用户痛点解析:你是否正遭遇这些网络管理困境…...
Qwen2.5-VL-7B-Instruct实操手册:对话历史自动保存+一键清空功能详解
Qwen2.5-VL-7B-Instruct实操手册:对话历史自动保存一键清空功能详解 1. 开篇:你的全能视觉助手来了 今天给大家介绍一个特别实用的工具——基于Qwen2.5-VL-7B-Instruct多模态大模型的视觉交互工具。这个工具专门为RTX 4090显卡优化过,用上了…...
