Error Flows in Go

原文链接:Error Flows in Go

作者:Preslav Rachev

下面这篇文章是我在 2023 年写的一篇关于 Go 中错误处理的文章的后续。建议还没读过这篇文章的人先看看这篇文章。

您想过 Go 应用程序出错时会发生什么吗?从 Go 应用程序的角度来看,发生错误有三个不同点:

  1. 错误 "originates ,",即首次出现(例如,您最后一次实际使用 errors.New)。
  2. 我们控制下的函数捕捉到错误,并决定采取以下措施之一
  • 缩短代码执行时间(panic, early return, clean up)
  • 或者更常见的做法是,将捕获到的错误与某些上下文进行封装,并将其返回给链中的前一个调用者。
  1. 错误会 "冒泡 "到最外层的调用者。除非途中发生了特殊情况(例如有人惊慌失措),否则这通常是 main 包,甚至是 main 函数本身。这通常是尝试窥探错误的地方,但老实说,在 99% 的情况下,这部分都很简单:

    1. 写出完整的错误信息,以便调试。
    2. 根据情况决定错误是否严重到需要继续运行应用程序。也许最好直接杀死应用程序,让基础架构重新启动。例如,如果错误发生在 HTTP 请求期间,这部分可能会向用户显示标准错误页面。但是,如果是在应用程序启动过程中发生的错误(例如,无法连接到数据库),我会直接 log.Fatal 该错误,然后让基础架构设置按顺序重启服务。

根据经验,除非您正在构建一个库,否则您可能不会花很多时间为您的应用程序引入新的错误。与其他 Go 应用程序一样,你的应用程序要么直接使用标准库,要么依赖于使用标准库的程序库。因此,在使用文件或数据库连接时,处理现有错误的几率要远远高于应用程序需要凭空编造新错误的几率。

不信?看看 Kubernetes 代码库中 errors.New 出现的次数与 fmt.Error(f) 出现的次数。多出两倍还多?

这就是我认为许多围棋开发者最大的误区。不,我说的不是这个:

go 复制代码
if err != nil {
	return err
}

这简直毫无用处。我强调过用有用的上下文来包装所有错误的重要性,我仍然坚定地坚持这一点。我认为人们弄错的是上下文中的信息传递部分。我经常看到信息试图告诉我哪里出错了。相反,他们应该告诉我在事情搞砸之前他们试图做什么。

事实上,如果你们发现自己处于第 1 阶段,并返回了一个全新的错误,那么使用 "**** 出错了 "的说法是没有问题的:

go 复制代码
return fmt.Errorf("DB failure: %w", err)

但你上面的 10 位调用者也会这样做。最后,不幸的值班开发人员得到的信息是这样的:

vbnet 复制代码
ERROR: DB failure: DB failure: DB failure: failed connecting to the DB: DB failure: "DB connection failed"

一点帮助都没有。你为什么不换个说法,告诉我代码在失败前试图做什么?换句话说,如果你在第 2 步,不要使用 "s*** got wrong"的模式,而是使用信息量更大的 "做了什么什么"。每个人都知道,如果他们看到错误日志,那是因为出错了,所以他们会多次感谢你没有一次又一次地使用 "failed"之类的词。

您可以查看我上一篇文章中的这一部分,以获得很好的说明。

相关推荐
邓熙榆13 分钟前
Logo语言的网络编程
开发语言·后端·golang
羊小猪~~4 小时前
MYSQL学习笔记(四):多表关系、多表查询(交叉连接、内连接、外连接、自连接)、七种JSONS、集合
数据库·笔记·后端·sql·学习·mysql·考研
ByteBlossom6666 小时前
MDX语言的语法糖
开发语言·后端·golang
计算机学姐6 小时前
基于微信小程序的驾校预约小程序
java·vue.js·spring boot·后端·spring·微信小程序·小程序
沈霁晨7 小时前
Ruby语言的Web开发
开发语言·后端·golang
DanceDonkey7 小时前
@RabbitListener处理重试机制完成后的异常捕获
开发语言·后端·ruby
平凡的运维之路8 小时前
vsftpd虚拟用户部署
后端
叫我:松哥8 小时前
基于Python django的音乐用户偏好分析及可视化系统设计与实现
人工智能·后端·python·mysql·数据分析·django
Leaf吧11 小时前
springboot 配置多数据源以及动态切换数据源
java·数据库·spring boot·后端
代码驿站52011 小时前
JavaScript语言的软件工程
开发语言·后端·golang