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

在Go中创建自定义错误

引言

Go提供了两种在标准库中创建错误的方法,[errors.Newfmt.Errorf],当与用户交流更复杂的错误信息时,或在调试时与未来的自己交流时,有时这两种机制不足以充分捕获和报告所发生的情况。为了传达更复杂的错误信息并实现更多的功能,我们可以实现标准库接口类型,error

其语法如下:

type error interface {Error() string
}

builtin包将error定义为一个接口,它只有一个Error()方法,返回一个字符串形式的错误消息。通过实现这个方法,我们可以将定义的任何类型转换为我们自己的error。

让我们尝试运行以下示例来查看error接口的实现:

package mainimport ("fmt""os"
)type MyError struct{}func (m *MyError) Error() string {return "boom"
}func sayHello() (string, error) {return "", &MyError{}
}func main() {s, err := sayHello()if err != nil {fmt.Println("unexpected error: err:", err)os.Exit(1)}fmt.Println("The string:", s)
}
Outputunexpected error: err: boom
exit status 1

在这里,我们创建了一个新的空结构类型MyError,并在其上定义了Error()方法。Error()方法返回字符串"boom"

main()中,我们调用函数sayHello,该函数返回一个空字符串和一个新的MyError实例。由于sayHello总是会返回错误,所以main()中的if语句体中的fmt.Println调用总是会执行。然后,我们使用fmt.Println打印短前缀字符串"unexpected error:"以及err变量中保存的MyError实例。

请注意,我们不必直接调用Error(),因为fmt包能够自动检测这是Error 的实现。它透明地调用Error()来获取字符串"boom",并将其与前缀字符串"unexpected Error: err:"连接起来。

在自定义错误中收集详细信息

有时候,自定义错误是捕获详细错误信息的最简洁方式。例如,假设我们想要捕获HTTP请求产生的错误的状态码;运行以下程序来查看error的实现,它允许我们清晰地捕获信息:

package mainimport ("errors""fmt""os"
)type RequestError struct {StatusCode intErr error
}func (r *RequestError) Error() string {return fmt.Sprintf("status %d: err %v", r.StatusCode, r.Err)
}func doRequest() error {return &RequestError{StatusCode: 503,Err:        errors.New("unavailable"),}
}func main() {err := doRequest()if err != nil {fmt.Println(err)os.Exit(1)}fmt.Println("success!")
}
Outputstatus 503: err unavailable
exit status 1

在这个例子中,我们创建了一个新的RequestError实例,并使用标准库中的errors.New函数提供状态码和一个错误。然后像前面的例子一样,我们使用fmt.Println打印它。

RequestErrorError()方法中,我们使用fmt.Sprintf函数来使用创建错误时提供的信息来构造一个字符串。

类型断言和自定义错误

error接口只公开了一个方法,但我们可能需要访问error实现的其他方法来正确处理错误。例如,我们可能有几个临时的error自定义实现,可以通过存在的 temporary()方法进行检索。

接口为类型提供了更广泛的方法集合,因此我们必须使用类型断言来更改view正在显示的方法,或者完全删除它。

下面的例子在之前的RequestError的基础上增加了一个Temporary()方法,该方法将表明调用者是否应该重试请求:

package mainimport ("errors""fmt""net/http""os"
)type RequestError struct {StatusCode intErr error
}func (r *RequestError) Error() string {return r.Err.Error()
}func (r *RequestError) Temporary() bool {return r.StatusCode == http.StatusServiceUnavailable // 503
}func doRequest() error {return &RequestError{StatusCode: 503,Err:        errors.New("unavailable"),}
}func main() {err := doRequest()if err != nil {fmt.Println(err)re, ok := err.(*RequestError)if ok {if re.Temporary() {fmt.Println("This request can be tried again")} else {fmt.Println("This request cannot be tried again")}}os.Exit(1)}fmt.Println("success!")
}
Outputunavailable
This request can be tried again
exit status 1

main()中,我们调用doRequest(),它会返回一个error接口给我们。我们首先打印 error()方法返回的错误消息。接下来,我们尝试使用类型断言re, ok := err.(*RequestError)来暴露RequestError中的所有方法。如果类型断言成功,那么我们使用Temporary()方法来查看此错误是否为临时错误。由于doRequest()设置的StatusCode503,这与http.StatusServiceUnavailable匹配,因此返回true并导致打印" this request can be try again"。在实践中,我们会发送另一个请求,而不是打印一条消息。

包装错误

通常,错误会在程序之外产生,例如:数据库、网络连接等。这些错误提供的错误消息不能帮助任何人找到错误的根源。在错误消息的开头使用额外的信息包装错误,可以为成功调试提供一些必要的上下文。

下面的例子演示了我们如何将一些上下文信息附加到从其他函数返回的晦涩的error上:

package mainimport ("errors""fmt"
)type WrappedError struct {Context stringErr     error
}func (w *WrappedError) Error() string {return fmt.Sprintf("%s: %v", w.Context, w.Err)
}func Wrap(err error, info string) *WrappedError {return &WrappedError{Context: info,Err:     err,}
}func main() {err := errors.New("boom!")err = Wrap(err, "main")fmt.Println(err)
}
Outputmain: boom!

WrappedError是一个有两个字段的结构体:一个是string类型的上下文消息,另一个是error类型,WrappedError提供了更多的信息。当Error()方法被调用时,我们再次使用fmt.Sprintf来打印上下文消息,然后Error(fmt.Sprintf也知道隐式调用Error()方法)。

main()中,我们使用errors.New创建一个错误,然后使用我们定义的wrap函数包装这个错误。这允许我们指出这个error是在"main"中生成的。此外,由于我们的WrappedError也是一个error,我们可以包装其他的WrappedError,这将允许我们看到一个链来帮助我们追踪错误的来源。在标准库的帮助下,我们甚至可以在错误中嵌入完整的堆栈跟踪。

总结

由于error接口只有一个方法,我们已经看到我们可以为不同的情况提供不同类型的错误。这可以涵盖从将多条信息作为错误的一部分进行沟通到实现指数回退的所有事情。虽然Go中的错误处理机制表面上看起来很简单,但我们可以使用这些自定义错误来处理常见和不常见的情况,从而实现相当丰富的处理。

Go还有另一种沟通意外行为的机制panics。在错误处理系列的下一篇文章中,我们将研究恐慌——它们是什么以及如何处理它们。

http://www.lryc.cn/news/198143.html

相关文章:

  • Vue.js2+Cesium1.103.0 十三、通过经纬度查询 GeoServer 发布的 wms 服务下的 feature 对象的相关信息
  • 使用STM32怎么喂狗 (IWDG)
  • GEE:计算和打印GEE程序的执行时间
  • GDPU 数据结构 天码行空5
  • SQLAlchemy学习-12.查询之 order_by 按desc 降序排序
  • 如何轻松打造数字人克隆系统+直播系统?OEM教你快速部署数字人SaaS系统源码
  • 药物滥用第四篇介绍
  • Apache Doris (四十三): Doris数据更新与删除 - Update数据更新
  • 面试算法29:排序的循环链表
  • python中不可变类型和可变类型
  • vue3封装Axios库的 API 请求并使用拦截器来处理请求和响应
  • RK3588开发笔记(二):基于方案商提供sdk搭建引入mpp和sdk的宿主机交叉编译Qt5.12.10环境
  • rust学习——函数返回值
  • 【Cadence】配置文件cdsinit和cdsenv的使用
  • 软考 系统架构设计师系列知识点之基于架构的软件开发方法ABSD(6)
  • MATLAB常用命令大全,非常详细(持续更新中)
  • js笔试面试题5道附答案
  • 4-k8s-部署springboot项目简单实践
  • Ai数字人直播系统SaaS源码大开源,源码独立部署助力中小企业发展!
  • 新的 Work Node 如何加入 K8s 集群 - Kubeadm ?
  • laravel框架的优缺点是什么?
  • 程序员接单都在用这六大平台,你呢?
  • 2022年亚太杯APMCM数学建模大赛D题储能系统中传热翅片的结构优化求解全过程文档及程序
  • 图像处理软件Photoshop 2023 mac新增功能 ps 2023中文版
  • CSS基本讲解与使用(详解)
  • 最新AI创作系统ChatGPT源码+搭建部署教程+支持GPT4.0+支持ai绘画(Midjourney)/支持Prompt
  • Linux系统之部署SSCMS内容管理系统并实现外网访问
  • JVS-rules中的基础与复合变量:规则引擎的心脏
  • RN:指定模拟器启动
  • 【ARM Cache 系列文章 10 -- ARM Cortex-A720 Hunter 介绍】