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

完美的错误处理:Go 语言最佳实践分享

Go 语言是一门非常流行的编程语言,由于其高效的并发编程和出色的网络编程能力,越来越受到广大开发者的青睐。在任何编程语言中,错误处理都是非常重要的一环,它关系到程序的健壮性和可靠性。Go 语言作为一门现代化的编程语言,自然也有其独特的错误处理机制。在本文中,我们将深入探讨 Go 语言中的错误处理机制,包括错误的基本概念、错误处理的基本方法、错误封装和自定义错误类型等方面,帮助读者更好地理解和掌握 Go 语言的错误处理技巧。

1. 错误的基本概念

在任何编程语言中,错误处理都需要我们首先理解错误的基本概念。在 Go 语言中,错误通常是一个接口类型,该接口定义如下:

type error interface {Error() string
}

可以看到,该接口只包含一个 Error 方法,该方法返回一个字符串,表示错误的信息。因此,任何类型只要实现了该接口的 Error 方法,就可以被当作一个错误来处理。Go 语言中的标准库提供了 errors 包,该包提供了一个简单的错误实现,示例如下:

package errors
​
func New(text string) error {return &errorString{text}
}type errorString struct {s string
}func (e *errorString) Error() string {return e.s
}

可以看到,该包提供了一个 New 函数,该函数接收一个字符串参数,返回一个 error 接口类型的错误。该包还定义了一个私有的 errorString 类型,该类型实现了 error 接口的 Error 方法,表示一个简单的字符串错误。当我们需要返回一个简单的字符串错误时,可以使用该包提供的 New 函数。例如:

import "errors"func someFunc() error {return errors.New("something went wrong")
}

2. 错误类型

在 Go 语言中,error 是一个接口类型,它只有一个方法 Error(),返回一个字符串类型的错误消息。如果一个函数返回一个非空的 error 类型,则意味着该函数执行过程中发生了错误。

type error interface {Error() string
}

错误类型通常是内置类型 error,我们可以在标准库中找到它:

var (ErrInvalidParam = errors.New("invalid parameter")ErrNotFound     = errors.New("not found")ErrInternal     = errors.New("internal error")
)

在这个例子中,我们使用 errors.New() 函数来创建了三个错误值,这些错误值将被用于不同的错误情况。当我们在编写函数时需要返回错误时,可以返回一个这样的错误值。

3. 自定义错误类型

在 Go 语言中,我们也可以定义自己的错误类型。如果我们希望自己的错误类型可以包含更多的信息,或者需要提供一些特定的行为,那么自定义错误类型就非常有用。
自定义错误类型可以是任何类型,只要它实现了 error 接口即可。下面是一个自定义错误类型的示例:

type MyError struct {message stringcode    int
}func (e *MyError) Error() string {return fmt.Sprintf("%s (code=%d)", e.message, e.code)
}func processFile(filename string) error {return &MyError{"File not found", 404}
}func main() {err := processFile("test.txt")fmt.Printf("Error: %s\n", err)
}

在上面的示例中,我们定义了一个 MyError 类型,该类型包含一个消息和一个错误代码。我们还定义了一个 Error() 方法来满足 error 接口的要求。最后,在 processFile() 函数中,我们返回一个新的 MyError 对象。
在 main() 函数中,我们打印错误信息。由于 MyError 类型实现了 Error() 方法,因此我们可以直接打印错误对象,而无需使用 fmt.Sprintf() 函数。
自定义错误类型非常灵活,并且可以帮助我们更好地组织代码和处理错误。但是,在创建自定义错误类型时,我们需要遵循一些最佳实践:

错误类型应该清晰地描述错误的类型和原因。
错误类型应该与错误的语境相匹配。例如,如果我们正在编写一个网络应用程序,我们可以定义一些与 HTTP 状态码相关的错误类型。
如果我们需要在错误类型之间共享某些字段或方法,我们可以使用嵌入类型(embedded types)。

4. 错误处理

在 Go 中,我们通常使用 if 语句来检查函数或方法的返回值是否为错误。以下是一个示例:

package main
​
import ("fmt""os"
)func main() {file, err := os.Open("file.txt")if err != nil {fmt.Printf("Error: %s", err.Error())return}defer file.Close()// 在这里进行文件操作
}

在上面的示例中,我们使用 os 包中的 Open 函数打开文件 “file.txt”。如果该文件无法打开,则 Open 函数将返回一个错误值。我们使用 if 语句来检查是否存在错误,如果存在错误,则打印错误信息并返回。否则,我们使用 defer 语句来关闭文件句柄。

5. errors.Is 和 errors.As

在之前的版本中,要比较一个 error 是否和一个特定的错误相同,需要使用字符串进行判断,但这种方式并不可靠,因为有可能在不同的地方,同一个错误信息被表示为不同的字符串,这样的话使用字符串进行判断就会失效。而 Go 1.13 中引入的 errors.Is 和 errors.As 函数,就可以解决这个问题。
errors.Is 函数可以检查 error 链中是否包含了某个错误。它接受两个参数,第一个参数是要检查的错误,第二个参数是要匹配的错误。如果匹配成功,函数会返回 true,否则返回 false。示例代码如下:

package main
​
import ("errors""fmt"
)func main() {err := errors.New("Something went wrong")if errors.Is(err, errors.New("Something went wrong")) {fmt.Println("Matched error")} else {fmt.Println("Did not match error")}
}

上面的代码中,我们使用了 errors.Is 函数来检查 err 是否与 errors.New(“Something went wrong”) 相匹配,由于它们的错误信息都是相同的,因此这个函数会返回 true。
除了 errors.Is,Go 1.13 还引入了另外一个函数 errors.As。与 errors.Is 不同,errors.As 函数是用来获取 error 链中特定类型的错误的。它接受两个参数,第一个参数是要检查的错误,第二个参数是一个指针,指向一个变量,这个变量的类型就是我们要获取的错误的类型。如果找到了匹配的错误,函数会把这个错误赋值给这个变量,并返回 true,否则返回 false。示例代码如下:

package main
​
import ("errors""fmt"
)type myError struct {code intmsg  string
}func (e myError) Error() string {return fmt.Sprintf("Error with code %d: %s", e.code, e.msg)
}func main() {err := myError{code: 404, msg: "Page not found"}var targetErr myErrorif errors.As(err, &targetErr) {fmt.Printf("Matched error: %+v\n", targetErr)} else {fmt.Println("Did not match error")}
}

上面的代码中,我们定义了一个 myError 类型,它实现了 Error 方法。我们然后创建了一个这个类型的实例 err,并定义了一个 targetErr 变量。接着,我们使用 errors.As 函数来检查 err 是否与 targetErr 的类型相匹配。由于它们的类型相同,因此这个函数会返回 true,并把 err 赋值给 targetErr。

6. panic 和 recover

在 Go 中,panic 和 recover 是用于处理错误和异常的两个内置函数。panic 用于引发一个 panic,这通常意味着一个严重的错误已经发生了,程序可能无法继续执行。recover 用于捕获 panic,以允许程序在 panic 后恢复执行或清理资源。

相关文章:

完美的错误处理:Go 语言最佳实践分享

Go 语言是一门非常流行的编程语言,由于其高效的并发编程和出色的网络编程能力,越来越受到广大开发者的青睐。在任何编程语言中,错误处理都是非常重要的一环,它关系到程序的健壮性和可靠性。Go 语言作为一门现代化的编程语言&#…...

vue首页多模块布局(标题布局)

<template><div class"box"><div class"content"><div class"box1" style"background-color: rgb(245,23,156)">第一个</div><div class"box2" style"background-color: rgb(12,233,…...

嵌入式系统>嵌入式硬件知识

AI芯片的特点包括 &#xff1a;新型计算范式AI芯片的关键特征&#xff1a; 1、新型的计算范式 AI 计算既不脱离传统计算&#xff0c;也具有新的计算特质&#xff0c;如处理的内容往往是非结构化数据&#xff08;视频、图片等&#xff09;。处理的过程通常需要很大的计算量&am…...

LeetCode 1402. 做菜顺序【排序,动态规划;贪心,前缀和,递推】1679

本文属于「征服LeetCode」系列文章之一&#xff0c;这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁&#xff0c;本系列将至少持续到刷完所有无锁题之日为止&#xff1b;由于LeetCode还在不断地创建新题&#xff0c;本系列的终止日期可能是永远。在这一系列刷题文章…...

【多线程】探索Java中的多线程编程

标题&#xff1a;探索Java中的多线程编程 摘要&#xff1a; Java是一种广泛使用的编程语言&#xff0c;具有强大的多线程编程能力。本文将深入探讨Java中的多线程编程&#xff0c;包括线程的创建、同步与互斥、线程池的使用以及常见的多线程编程模式。通过示例代码和详细解释&…...

【算法题】翻转对

题目&#xff1a; 给定一个数组 nums &#xff0c;如果 i < j 且 nums[i] > 2*nums[j] 我们就将 (i, j) 称作一个重要翻转对。 你需要返回给定数组中的重要翻转对的数量。 示例 1: 输入: [1,3,2,3,1] 输出: 2 示例 2: 输入: [2,4,3,5,1] 输出: 3 注意: 给定数组的长…...

适用于 Mac 或 Windows 的 4 种最佳 JPEG/PNG图片 恢复软件

您的计算机或外部存储驱动器上很可能有大量 JPEG /PNG图片照片&#xff0c;但不知何故&#xff0c;您意识到一些重要的 JPEG /PNG图片文件丢失或被删除&#xff0c;它们对您来说意义重大&#xff0c;您想要找回它们. 4 种最佳 JPEG/PNG图片 恢复软件 要成功执行 JPEG /PNG图片…...

位置信息API

位置信息API 一、获取当前位置&#xff1a;wx.getLocation(object)二、选择位置&#xff1a;wx.chooseLocation(object)三、打开位置&#xff1a;wx.openLocation(object)四、监听位置事件五、地图组件控制API六、收货地址API&#xff1a;wx.chooseAddress(object) 一、获取当前…...

MySQL——九、SQL编程

MySQL 一、触发器1、触发器简介2、创建触发器3、一些常见示例 二、存储过程1、什么是存储过程或者函数2、优点3、存储过程创建与调用 三、存储函数1、存储函数创建和调用2、修改存储函数3、删除存储函数 四、游标1、声明游标2、打开游标3、使用游标4、关闭游标游标案例 一、触发…...

threejs(4)-纹理材质高级操作

一、纹理重复_缩放_旋转_位移操作 // 导入threejs import * as THREE from "three"; // 导入轨道控制器 import { OrbitControls } from "three/examples/jsm/controls/OrbitControls.js"; // 导入lil.gui import { GUI } from "three/examples/jsm/l…...

Redis | 数据结构(01)

这里写自定义目录标题 Redis 速度快的原因除了它是内存数据库&#xff0c;使得所有的操作都在内存上进行之外&#xff0c;还有一个重要因素&#xff0c;它实现的数据结构&#xff0c;使得我们对数据进行增删查改操作时&#xff0c;Redis 能高效的处理。 因此&#xff0c;这次我…...

一文详解多模态大模型发展及高频因子计算加速GPU算力 | 英伟达显卡被限,华为如何力挽狂澜?

★深度学习、机器学习、多模态大模型、深度神经网络、高频因子计算、GPT-4、预训练语言模型、Transformer、ChatGPT、GenAI、L40S、A100、H100、A800、H800、华为、GPU、CPU、英伟达、NVIDIA、卷积神经网络、Stable Diffusion、Midjourney、Faster R-CNN、CNN 随着人工智能技术…...

debian 10 安装apache2 zabbix

nginx 可以略过&#xff0c;改为apache2 apt updateapt-get install nginx -ynginx -v nginx version: nginx/1.14.2mysql 安装参考linux debian10 安装mysql5.7_debian apt install mysql5.7-CSDN博客 Install and configure Zabbix for your platform a. Install Zabbix re…...

Qt之菜单栏、工具栏、状态栏介绍及工具栏QAction的动态增删显示实现方式

目的 端应用程序或者编辑器基本都支持工具栏快捷功能的动态增删&#xff0c;即通过在菜单栏上打钩就可以在工具栏上看到相应功能的快捷按钮&#xff0c;取消打钩则在工具栏上就移除了该功能的快捷按钮。那么Qt如何实现这个功能&#xff0c;本篇目的就是记录实现此功能的方法及思…...

十四天学会C++之第八天:文件操作

1. 文件的打开和关闭 文件操作的基本概念。打开文件&#xff1a;使用fstream库打开文件以供读写。关闭文件&#xff1a;确保文件在使用完毕后正确关闭。 文件的打开和关闭&#xff1a;C 文件操作入门 在C编程中&#xff0c;文件操作是一项重要的任务&#xff0c;可以读取和写…...

基于(N-1)×(N-1)棋盘的解的情况推出N×N棋盘的解的情况的N皇后问题

N皇后问题是一个比较经典的问题&#xff0c;其主要目标是在NN的棋盘上&#xff0c;放置N个皇后&#xff0c;要求所有皇后之间不能互相攻击&#xff0c;即任意两个皇后不能处在同一行、同一列或同一对角线上。解决该问题可以采用递归的方式&#xff0c;基于(N-1)棋盘的解的情况推…...

Vue mixin混入

可以把多个组件中共有的配置提取出来构成一个混入。 一、配置混入 &#xff08;一&#xff09; 创建mixin.js 这里的名字可以自定义&#xff0c;但是为了方便识别&#xff0c;多数场景下都写mixin。 mixin.js 要创建在src目录下&#xff0c;与main.js平级&#xff1a; &…...

基于 FFmpeg 的跨平台视频播放器简明教程(十):在 Android 运行 FFmpeg

系列文章目录 基于 FFmpeg 的跨平台视频播放器简明教程&#xff08;一&#xff09;&#xff1a;FFMPEG Conan 环境集成基于 FFmpeg 的跨平台视频播放器简明教程&#xff08;二&#xff09;&#xff1a;基础知识和解封装&#xff08;demux&#xff09;基于 FFmpeg 的跨平台视频…...

正点原子嵌入式linux驱动开发——Linux LCD驱动

LCD是很常用的一个外设&#xff0c;通过LCD可以显示绚丽的图片、界面等&#xff0c;提交人机交互的效率。STM32MP1提供了一个LTDC接口用于连接RGB接口的液晶屏。本章就来学校一下如何在Linux下驱动LCD屏。 LCD和LTDC简介 LCD简介 这里在当时学习stm32裸机开发的时候就学过了…...

2-Java进阶知识总结-6-多线程

文章目录 多线程--基本概念并发和并行进程和线程多线程 多线程--实现方式一&#xff0c;继承Thread类方法介绍实现步骤注意事项 方式二&#xff0c;实现Runnable接口Thread构造方法实现步骤 方式三&#xff0c;实现Callable接口方法介绍实现步骤 三种多线程实现方法对比 多线程…...

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…...

eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)

说明&#xff1a; 想象一下&#xff0c;你正在用eNSP搭建一个虚拟的网络世界&#xff0c;里面有虚拟的路由器、交换机、电脑&#xff08;PC&#xff09;等等。这些设备都在你的电脑里面“运行”&#xff0c;它们之间可以互相通信&#xff0c;就像一个封闭的小王国。 但是&#…...

使用分级同态加密防御梯度泄漏

抽象 联邦学习 &#xff08;FL&#xff09; 支持跨分布式客户端进行协作模型训练&#xff0c;而无需共享原始数据&#xff0c;这使其成为在互联和自动驾驶汽车 &#xff08;CAV&#xff09; 等领域保护隐私的机器学习的一种很有前途的方法。然而&#xff0c;最近的研究表明&…...

理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端

&#x1f31f; 什么是 MCP&#xff1f; 模型控制协议 (MCP) 是一种创新的协议&#xff0c;旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议&#xff0c;它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...

电脑插入多块移动硬盘后经常出现卡顿和蓝屏

当电脑在插入多块移动硬盘后频繁出现卡顿和蓝屏问题时&#xff0c;可能涉及硬件资源冲突、驱动兼容性、供电不足或系统设置等多方面原因。以下是逐步排查和解决方案&#xff1a; 1. 检查电源供电问题 问题原因&#xff1a;多块移动硬盘同时运行可能导致USB接口供电不足&#x…...

Matlab | matlab常用命令总结

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

【python异步多线程】异步多线程爬虫代码示例

claude生成的python多线程、异步代码示例&#xff0c;模拟20个网页的爬取&#xff0c;每个网页假设要0.5-2秒完成。 代码 Python多线程爬虫教程 核心概念 多线程&#xff1a;允许程序同时执行多个任务&#xff0c;提高IO密集型任务&#xff08;如网络请求&#xff09;的效率…...

GC1808高性能24位立体声音频ADC芯片解析

1. 芯片概述 GC1808是一款24位立体声音频模数转换器&#xff08;ADC&#xff09;&#xff0c;支持8kHz~96kHz采样率&#xff0c;集成Δ-Σ调制器、数字抗混叠滤波器和高通滤波器&#xff0c;适用于高保真音频采集场景。 2. 核心特性 高精度&#xff1a;24位分辨率&#xff0c…...

使用 SymPy 进行向量和矩阵的高级操作

在科学计算和工程领域&#xff0c;向量和矩阵操作是解决问题的核心技能之一。Python 的 SymPy 库提供了强大的符号计算功能&#xff0c;能够高效地处理向量和矩阵的各种操作。本文将深入探讨如何使用 SymPy 进行向量和矩阵的创建、合并以及维度拓展等操作&#xff0c;并通过具体…...

Angular微前端架构:Module Federation + ngx-build-plus (Webpack)

以下是一个完整的 Angular 微前端示例&#xff0c;其中使用的是 Module Federation 和 npx-build-plus 实现了主应用&#xff08;Shell&#xff09;与子应用&#xff08;Remote&#xff09;的集成。 &#x1f6e0;️ 项目结构 angular-mf/ ├── shell-app/ # 主应用&…...