Go的错误处理
什么是错误?
错误表示程序中发生的任何异常情况。假设我们正在尝试打开一个文件,但该文件在文件系统中不存在。这是一种异常情况,表示为错误。
Go 中的错误是普通的旧值。就像任何其他内置类型(例如 int、float64 等)一样,错误值可以存储在变量中、作为参数传递给函数、从函数返回等。
错误使用内置error
类型表示。我们将在本教程后面详细了解该error
类型。
例子
让我们立即开始尝试打开一个不存在的文件的示例程序。
package mainimport ( "fmt""os"
)func main() { f, err := os.Open("/test.txt")if err != nil {fmt.Println(err)return}fmt.Println(f.Name(), "opened successfully")
}
Run in playground
在上面程序的第 9 步中,我们尝试打开路径中的文件/test.txt
(该文件显然不会存在于 Playground 中)。包的*Open*函数os
具有以下签名,
*func Open(名称字符串) (File, error)
如果文件已成功打开,则 Open 函数将返回文件处理程序,错误将为 nil。如果打开文件时出现错误,将返回非零错误。
如果函数或方法返回错误,那么按照惯例,它必须是函数返回的最后一个值。因此该Open
函数返回error
最后一个值。
**Go 中处理错误的惯用方法是将返回的错误与nil
. nil 值表示没有发生错误,非 nil 值表示存在错误。**在我们的例子中,
我们检查错误是否不等于nil
。如果不是nil
,我们只需打印错误并从主函数返回。
运行该程序将打印
open /test.txt: No such file or directory
完美😃。我们收到一条错误消息,指出该文件不存在。
错误类型表示
让我们更深入地研究一下内置error
类型是如何定义的。error是具有以下定义的接口类型
type error interface { Error() string
}
它包含一个带有签名的方法Error() string
。任何实现此接口的类型都可以用作错误。此方法提供错误的描述。
当打印错误时,fmt.Println
函数内部调用该Error() string
方法来获取错误的描述打印方式
从错误中提取更多信息的不同方法
现在我们知道了error
是一个接口类型,让我们看看如何提取有关错误的更多信息。
在上面的示例中,我们刚刚打印了错误的描述。如果我们想要导致错误的文件的实际路径怎么办?获取此信息的一种可能方法是解析错误字符串。这是我们程序的输出,
open /test.txt: No such file or directory
我们可以解析此错误消息并获取导致错误的文件的文件路径“/test.txt”,但这是一种肮脏的做法。在较新版本的 Go 中,错误描述可能随时更改,我们的代码将会崩溃。
有没有更好的方法来获取文件名🤔?答案是肯定的,这是可以做到的,并且 Go 标准库使用不同的方式来提供有关错误的更多信息。让我们一一看看。
1. 将错误转换为基础类型并从结构体字段中检索更多信息
如果你仔细阅读Open函数的文档,你会发现它返回一个类型为*PathError.
PathError的错误,它是一个结构体类型,它在标准库中的实现如下:
type PathError struct { Op stringPath stringErr error
}func (e *PathError) Error() string { return e.Op + " " + e.Path + ": " + e.Err.Error() }
如果您有兴趣知道上述源代码存在于哪里,可以在这里找到https://cs.opensource.google/go/go/+/refs/tags/go1.19:src/io/fs/fs .go;l=250
从上面的代码中,你可以明白,是通过声明方法*PathError
来实现的。此方法将操作、路径和实际错误连接起来并返回。因此我们得到了错误消息,error interface``Error() string
open /test.txt: No such file or directory
Path
struct 字段 包含PathError
导致错误的文件的路径。
我们可以使用errors包中的As函数将错误转换为其基础类型。该As
函数的描述谈到了错误链。请暂时忽略它。我们将在单独的教程中了解错误链和包装的工作原理。
简单的描述As
是,它尝试将错误转换为错误类型,并返回 true 或 false 指示转换是否成功。
一个程序会让事情变得清晰。让我们修改上面编写的程序并使用该As
函数打印路径。
package mainimport ( "errors""fmt""os"
)func main() { f, err := os.Open("test.txt")if err != nil {var pErr *os.PathErrorif errors.As(err, &pErr) {fmt.Println("Failed to open file at path", pErr.Path)return}fmt.Println("Generic error", err)return}fmt.Println(f.Name(), "opened successfully")
}
Run in playground
在上面的程序中,我们首先检查错误是否不在nil
, 然后我们使用As
的函数。 转换err
为*os.PathError
.
如果转换成功,As
将返回true
。
如果您想知道为什么pErr
是指针,原因是错误接口是由指针实现的PathError
,因此pErr
是指针。下面的代码显示了*PathError
错误接口的实现。
func (e *PathError) Error() string { return e.Op + " " + e.Path + ": " + e.Err.Error() }
该As
函数要求第二个参数是指向实现错误的类型的指针。因此我们通过了&perr
。
该程序输出,
Failed to open file at path test.txt
如果底层错误不是*os.PathError
类型,则打印一般错误消息。
太棒了😃。我们已经成功地使用该As
函数从错误中获取文件路径。
2. 使用方法检索更多信息
从错误中获取更多信息的第二种方法是找出基础类型并通过调用结构类型上的方法来获取更多信息。
让我们通过一个例子更好地理解这一点。
标准库中的DNSError*结构*类型定义如下:
type DNSError struct { ...
}func (e *DNSError) Error() string { ...
}
func (e *DNSError) Timeout() bool { ...
}
func (e *DNSError) Temporary() bool { ...
}
该DNSError
结构有两个方法Timeout() bool
,Temporary()
它们返回一个布尔值,指示错误是由于超时还是临时错误。
让我们编写一个程序,将错误转换为*DNSError
类型,并调用上述方法来确定错误是暂时的还是由于超时造成的。
package mainimport ( "errors""fmt""net"
)func main() { addr, err := net.LookupHost("baidu12345.com")if err != nil {var dnsErr *net.DNSErrorif errors.As(err, &dnsErr) {if dnsErr.Timeout() {fmt.Println("operation timed out")return}if dnsErr.Temporary() {fmt.Println("temporary error")return}fmt.Println("Generic DNS error", err)return}fmt.Println("Generic error", err)return}fmt.Println(addr)
}
注意:DNS 查找在 Playground 中不起作用。请在您的本地计算机上运行该程序。
在上面的程序中,我们正在尝试获取无效域名的IP地址baidu123.com
。我们通过使用As
该函数并将其转换为DNSError
来 获取错误的基本值。然后我们分别在14和18行检查错误是由于超时还是暂时错误的。
在我们的例子中,错误既不是暂时的,也不是由于超时造成的,因此程序将打印:
Generic DNS error lookup baidu12345.com: no such host
如果错误是暂时的或者由于超时,那么相应的 if 语句就会执行,我们可以适当地处理它。
3. 直接比较
获取有关错误的更多详细信息的第三种方法是直接与 类型的变量进行比较error
。让我们通过一个例子来理解这一点。
包的Glob函数filepath
用于返回与某个模式匹配的所有文件的名称。当模式格式错误时,此ErrBadPattern
函数将返回错误。
ErrBadPattern在包中定义filepath
为全局变量。
var ErrBadPattern = errors.New("syntax error in pattern")
error.New() 用于创建一个新错误。我们将在下一个教程中详细讨论这一点。
当模式格式错误时,Glob 函数将返回ErrBadPattern 。
让我们编写一个小程序来检查此错误。
package mainimport ( "errors""fmt""path/filepath"
)func main() { files, err := filepath.Glob("[")if err != nil {if errors.Is(err, filepath.ErrBadPattern) {fmt.Println("Bad pattern error:", err)return}fmt.Println("Generic error:", err)return}fmt.Println("matched files", files)
}
Run in playground
在上面的程序中,我们搜索格式[
错误的模式文件。我们检查错误是否不为nil。为了获得有关错误的更多信息,我们直接使用Is函数将filepath.ErrBadPattern
inline 进行比较。与 类似As
,该Is
函数在错误链上工作。我们将在下一个教程中了解更多相关内容。
出于本教程的目的,如果传递给该函数的两个错误相同,则Is
可以将该函数视为true
然后返回。
Is
第 12 行返回 true 。因为错误是由于格式错误造成的。该程序将打印,
Bad pattern error: syntax error in pattern
标准库使用上述任何一种方法来提供有关错误的更多信息。我们将在下一个教程中使用这些方法来创建我们自己的自定义错误
不要忽视错误
永远不要忽视错误。忽略错误会招致麻烦。让我重写示例,其中列出了与模式匹配的所有文件的名称,忽略错误。
package mainimport ( "fmt""path/filepath"
)func main() { files, _ := filepath.Glob("[")fmt.Println("matched files", files)
}
Run in playground
从前面的例子中我们已经知道是无效的。我通过_
使用空白标识符忽略了函数返回的错误。我只是打印匹配的文件。该程序将打印
matched files []
由于我们忽略了该错误,因此输出看起来好像没有文件与该模式匹配,但实际上该模式本身格式错误。所以永远不要忽视错误。
本教程到此结束。
相关文章:

Go的错误处理
什么是错误? 错误表示程序中发生的任何异常情况。假设我们正在尝试打开一个文件,但该文件在文件系统中不存在。这是一种异常情况,表示为错误。 Go 中的错误是普通的旧值。就像任何其他内置类型(例如 int、float64 等)…...

云原生相关概念
云计算 指托管在外部数据中心并按使用量付费提供给用户的软件基础设施。公司不必为昂贵的服务器付费并进行维护。相反,他们可以使用云提供商提供的按需云原生服务,例如存储、数据库和分析。 云原生 是在 云计算环境 中构建、部署和管理现代应用程序的…...

【JS】this指向
一、this指向的四种规则 1.默认绑定规则 默认指向:指向window 独立调用:指向window 对象指向,比较的是引用地址。 console.log(this window); //true console.log({} {}); //false //函数的独立调用 function test(){console.lo…...

SpringCloud Alibaba Demo(Nacos,OpenFeign,Gatway,Sentinel)
开源地址: ma/springcloud-alibaba-demo 简介 参考:https://www.cnblogs.com/zys2019/p/12682628.html SpringBoot、SpringCloud 、SpringCloud Alibaba 以及各种组件存在版本对应关系。可参考下面 版本对应 项目前期准备 启动nacos. ./startup.c…...

基于nodejs+vue畅听校园点歌系统的设计与实现
目 录 摘 要 I ABSTRACT II 目 录 II 第1章 绪论 1 1.1背景及意义 1 1.2 国内外研究概况 1 1.3 研究的内容 1 第2章 相关技术 3 2.1 nodejs简介 4 2.2 express框架介绍 6 2.4 MySQL数据库 4 第3章 系统分析 5 3.1 需求分析 5 3.2 系统可行性分析 5 3.2.1技术可行性:…...

IDEA 设置代码注释模板
功能简介: 每次看别人代码时,面对毫无注释的类,除了头大还是头大, 以下提供了一种代码类注释模板 新建java类的时候,自动增加类注释,养成代码开发好习惯 效果展示: 代码模板: #if (…...

emoji对齐 特殊字符对齐 文本对齐
emoji如何对齐 特殊字符如何对齐 高级文本对齐 问题引出 我们在程序打印输出时,如何我们所输出的字符中包含emoji文本,或者其它特殊的字符文本,则我们的打印对齐效果将出现错位。以下代码复现了这一效果(tips: 马老师…...

Selenium Python 中的动作链
Selenium 是一个用于自动化的独立的基于 Web 的工具。 它是任何人都可以使用的开源工具。 与Python语言结合使用该工具进行测试。 操作链是 Selenium 的基本组成部分,提供了一种管理低级交互的方法,例如按键、鼠标移动、鼠标按钮操作以及与上下文菜单的…...

OceanBase:03-集群部署
目录 一、集群规划 二、配置要求 三、部署前配置 1.配置 limits.conf 2.配置 sysctl.conf 3.关闭防火墙 4.关闭 SELinux 5.创建数据目录,修改文件所有者信息 6.设置无密码 SSH 登录 7.安装jdk 四、解压执行安装 五、集群部署 1.OBD命令行部署 2. OBD白…...

PTA: 矩阵的乘法运算
矩阵的乘法运算 题目输入格式输出格式输入样例输出样例 代码 题目 线性代数中的矩阵可以表示为一个row*column的二维数组,当row和column均为1时,退化为一个数,当row为1时,为一个行向量,当column为1时&…...

4K Video Downloader Pro v4.28.0(视频下载器)
4K Video Downloader Pro是一款专业的视频下载软件,支持从YouTube、Vimeo、Facebook、Instagram、TikTok等主流视频网站下载高质量的4K、HD和普通视频。它的操作流程简单,只需复制视频链接并粘贴到软件中即可开始下载。此外,该软件还提供了多…...

java pdf,word,ppt转图片
pom.xml <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://maven.apache.org/POM/4.0.0…...

map set
目录 一、关联式容器 二、键值对 三、树形结构的关联式容器 3.1 set 3.1.1 set的介绍 3.1.2 set的使用 3.2 multiset 3.2.1 multiset的介绍 3.2.2 multiset的使用 3.3 map 3.3.1 map的介绍 3.3.2 map的使用 …...
Fourier分析导论——第3章——Fourier级数的收敛性(E.M. Stein R. Shakarchi)
第 3 章 Fourier级数的收敛性(Convergence of Fourier Series) The sine and cosine series, by which one can represent an arbitrary function in a given interval, enjoy among other remarkable properties that of being convergent. This property did not escape…...

解决ruoyi-vue部署到域名子路径静态资源404
参考ruoyi前端手册...

游戏引擎中为什么要用四元数表示旋转而不用欧拉角旋转?
个人观点,仅供参考,如有错误可太刺激了 四元数的简单概念和使用 欧拉角通常用于表示一个物体的旋转状态,而不是表示旋转过程。 欧拉角描述的是物体相对于某个参考坐标系的朝向或旋转状态,通常以不同的轴(例如&#x…...

E-Office(泛微OA)前台任意文件读取漏洞复现
简介 泛微E-Office是一款企业级的全流程办公自动化软件,它包括协同办公、文档管理、知识管理、工作流管理等多个模块,涵盖了企业日常工作中的各个环节。在该产品前台登录页存在文件读取漏洞。 officeserver.php文件存在任意文件读取漏洞,通…...

前端小案例 | 喵喵大王立大功 | 一个带便利贴功能的todolist面板
文章目录 📚html📚css📚js🐇stickynote.js🐇todolist.js🐇clock.js 📚html <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><m…...

算法训练营第十一天 | 20. 有效的括号、 1047. 删除字符串中的所有相邻重复项、150. 逆波兰表达式求值
目录: 力扣 20. 有效的括号力扣 1047. 删除字符串中的所有相邻重复项力扣 150. 逆波兰表达式求值 问题一、 20. 有效的括号 题目链接:20. 有效的括号 - 力扣(LeetCode) 思路分析: 很多朋友刚开始接触这一类题的时候…...

Python unittest单元测试框架 TestSuite测试套件
TestSuite 测试套件简介 对一个功能的验证往往是需要很多多测试用例,可以把测试用例集合在一起执行,这就产生了测试套件TestSuite 的概念,它是用来组装单个测试用例,规定用例的执行的顺序,而且TestSuite也可以嵌套Tes…...

FSB逮捕为乌克兰网络部队工作的俄罗斯黑客
导语 近日,俄罗斯联邦安全局(FSB)逮捕了两名涉嫌协助乌克兰网络部队对俄罗斯重要基础设施目标进行网络攻击的个人。这起事件引起了广泛关注,涉及到了网络安全和国际关系等多个领域。本文将为您详细介绍这一事件的背景和最新进展。…...

【PC电脑windows-学习样例tusb_serial_device-ESP32的USB模拟串口程序+VScode建立工程+usb组件添加+-基础样例学习】
【PC电脑windows-学习样例tusb_serial_device-ESP32的USB模拟串口程序-基础样例学习】 1、概述2、实验环境3-1、 物品说明3-2、所遇问题:ESP32 cannot open source file "tinyusb.h"或者“tinyusb.h:No such file or directory ....”3-3、解决问题&#…...

LeetCode75——Day26
文章目录 一、题目二、题解 一、题目 394. Decode String Given an encoded string, return its decoded string. The encoding rule is: k[encoded_string], where the encoded_string inside the square brackets is being repeated exactly k times. Note that k is guar…...

面试算法53:二叉搜索树的下一个节点
题目 给定一棵二叉搜索树和它的一个节点p,请找出按中序遍历的顺序该节点p的下一个节点。假设二叉搜索树中节点的值都是唯一的。例如,在图8.9的二叉搜索树中,节点8的下一个节点是节点9,节点11的下一个节点是null。 分析…...

2023SHCTF web方向wp
1.ezphp 看一眼,你大爷,啥玩意都给我过滤完了。 还好下面有preg_replace()/e,会把replacement当作php语句执行 传参pattern.*, .*表示任意字符,code{${phpinfo()}} ,为什么这样写,因为,print_…...

从物理磁盘到数据库 —— 存储IO链路访问图
原图来自:数据库IO链路访问图 – OracleBlog 由于很复杂,为了加深理解自己重新画了一次,另外参考其他文档补充了各部分的插图和介绍。 一、 存储服务器 1. 物理磁盘 外层的壳子称为硬盘笼 cage 2. chunklet Chunklet 是一个虚拟概念而不是实…...

基于java+springboot+vue在线选课系统
项目介绍 本系统结合计算机系统的结构、概念、模型、原理、方法,在计算机各种优势的情况下,采用JAVA语言,结合SpringBoot框架与Vue框架以及MYSQL数据库设计并实现的。员工管理系统主要包括个人中心、课程管理、专业管理、院系信息管理、学生…...

GO学习之 同步操作sync包
GO系列 1、GO学习之Hello World 2、GO学习之入门语法 3、GO学习之切片操作 4、GO学习之 Map 操作 5、GO学习之 结构体 操作 6、GO学习之 通道(Channel) 7、GO学习之 多线程(goroutine) 8、GO学习之 函数(Function) 9、GO学习之 接口(Interface) 10、GO学习之 网络通信(Net/Htt…...

NUUO网络摄像头(NVR)RCE漏洞复现
简介 NUUO Network Video Recorder(NVR)是中国台湾NUUO公司的一款网络视频记录器。 NUUO NVR视频存储管理设备的__debugging_center_utils___.php文件存在未授权远程命令执行漏洞,攻击者可在没有任何权限的情况下通过log参数执行任意命令。…...

一款快速获取目标网站关键信息的工具
1.摘要 今天要介绍的这款工具是一个快速收集网站信息的开源脚本, 采用Python语言编写, 该工具可以快速收集网站的页面标题、网站上次更新日期、DNS信息、子域、防火墙名称、网站使用的技术栈、证书等信息, 默认支持对验证码和JavaScript内容执行绕过操作。 2.工具安装使用 使…...