一些关于Go程序错误处理的相关建议
本篇文章向大家介绍《一些关于Go程序错误处理的相关建议》,主要包括处理、go错误,具有一定的参考价值,需要的朋友可以参考一下。
比如,有人写代码就会在每一层都判断错误并记录日志,从代码层面看,貌似很严谨,但是如果看日志会发现一堆重复的信息,等到排查问题时反而会造成干扰。
今天给大家总结三点Go代码错误处理相关的最佳实践给大家。
这些最佳实践也是网上一些前辈分享的,我自己实践后在这里用自己的语言描述出来,希望能对大家有所帮助。
认识error
Go程序通过error类型的值表示错误
error类型是一个内建接口类型,该接口只规定了一个返回字符串值的Error方法。
type error interface { Error() string }
Go语言的函数经常会返回一个error值,调用者通过测试error值是否是nil来进行错误处理。
i, err := strconv.Atoi("42") if err != nil { fmt.Printf("couldn't convert number: %v\n", err) return } fmt.Println("Converted integer:", i)
error为nil时表示成功;非nil的error表示失败。
自定义错误记得要实现error接口
我们经常会定义符合自己需要的错误类型,但是记住要让这些类型实现error接口,这样就不用在调用方的程序里引入额外的类型。
比如下面我们自己定义了myError这个类型,如果不实现error接口的话,调用者的代码中就会被myError这个类型侵入。比如下面的run函数,在定义返回值类型时,直接定义成error即可。
package myerror import ( "fmt" "time" ) type myError struct { Code int When time.Time What string } func (e *myError) Error() string { return fmt.Sprintf("at %v, %s, code %d", e.When, e.What, e.Code) } func run() error { return &MyError{ 1002, time.Now(), "it didn't work", } } func TryIt() { if err := run(); err != nil { fmt.Println(err) } }
如果myError不实现error接口的话,这里的返回值类型就要定义成myError类型。可想而知,紧接着调用者的程序里就要通过myError.Code == xxx 来判断到底是哪种具体的错误(当然想要这么干得先把myError改成导出的MyError)。
那调用者判断自定义error是具体哪种错误的时候应该怎么办呢,myError并未向包外暴露,答案是通过向包外暴露检查错误行为的方法来实现。
myerror.IsXXXError(err) ...
抑或是通过比较error本身与包向外暴露的常量错误是否相等来判断,比如操作文件时常用来判断文件是否结束的io.EOF。
类似的还有gorm.ErrRecordNotFound等各种开源包对外暴露的错误常量。
if err != io.EOF { return err }
错误处理常犯的错误
先看一段简单的程序,看大家能不能发现一些细微的问题
func WriteAll(w io.Writer, buf []byte) error { _, err := w.Write(buf) if err != nil { log.Println("unable to write:", err) // annotated error goes to log file return err // unannotated error returned to caller } return nil } func WriteConfig(w io.Writer, conf *Config) error { buf, err := json.Marshal(conf) if err != nil { log.Printf("could not marshal config: %v", err) return err } if err := WriteAll(w, buf); err != nil { log.Println("could not write config: %v", err) return err } return nil } func main() { err := WriteConfig(f, &conf) fmt.Println(err) // io.EOF }
错误处理常犯的两个问题
上面程序的错误处理暴露了两个问题:
1.底层函数WriteAll在发生错误后,除了向上层返回错误外还向日志里记录了错误,上层调用者做了同样的事情,记录日志然后把错误再返回给程序顶层。
因此在日志文件中得到一堆重复的内容
unable to write: io.EOF
could not write config: io.EOF
...
2. 在程序的顶部,虽然得到了原始错误,但没有相关内容,换句话说没有把WriteAll、WriteConfig记录到 log 里的那些信息包装到错误里,返回给上层。
针对这两个问题的解决方案可以是,在底层函数WriteAll、WriteConfig中为发生的错误添加上下文信息,然后将错误返回上层,由上层程序最后处理这些错误。
一种简单的包装错误的方法是使用fmt.Errorf函数,给错误添加信息。
func WriteConfig(w io.Writer, conf *Config) error { buf, err := json.Marshal(conf) if err != nil { return fmt.Errorf("could not marshal config: %v", err) } if err := WriteAll(w, buf); err != nil { return fmt.Errorf("could not write config: %v", err) } return nil } func WriteAll(w io.Writer, buf []byte) error { _, err := w.Write(buf) if err != nil { return fmt.Errorf("write failed: %v", err) } return nil }
给错误附加上下文信息
fmt.Errorf只是给错误添加了简单的注解信息,如果你想在添加信息的同时还加上错误的调用栈,可以借助github.com/pkg/errors这个包提供的错误包装能力。
//只附加新的信息 func WithMessage(err error, message string) error //只附加调用堆栈信息 func WithStack(err error) error //同时附加堆栈和信息 func Wrap(err error, message string) error
有包装方法,就有对应的解包方法,Cause方法会返回包装错误对应的最原始错误--即会递归地进行解包。
func Cause(err error) error
下面是使用github.com/pkg/errors改写后的错误处理程序
func ReadFile(path string) ([]byte, error) { f, err := os.Open(path) if err != nil { return nil, errors.Wrap(err, "open failed") } defer f.Close() buf, err := ioutil.ReadAll(f) if err != nil { return nil, errors.Wrap(err, "read failed") } return buf, nil } func ReadConfig() ([]byte, error) { home := os.Getenv("HOME") config, err := ReadFile(filepath.Join(home, ".settings.xml")) return config, errors.WithMessage(err, "could not read config") } func main() { _, err := ReadConfig() if err != nil { fmt.Printf("original error: %T %v\n", errors.Cause(err), errors.Cause(err)) fmt.Printf("stack trace:\n%+v\n", err) os.Exit(1) } }
上面格式化字符串时用的 %+v 是在 % v 基础上,对值进行展开,即展开复合类型值,比如结构体的字段值等明细。
这样既能给错误添加调用栈信息,又能保留对原始错误的引用,通过Cause可以还原到最初始引发错误的原因。
总结
总结一下,错误处理的原则就是:
错误只在逻辑的最外层处理一次,底层只返回错误。
底层除了返回错误外,要对原始错误进行包装,增加错误信息、调用栈等这些有利于排查的上下文信息。
终于介绍完啦!小伙伴们,这篇关于《一些关于Go程序错误处理的相关建议》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

- 上一篇
- Go语言基础语法之结构体及方法详解

- 下一篇
- 小学生也能看懂的Golang异常处理recover panic
-
- 健壮的身影
- 太详细了,码住,感谢师傅的这篇技术贴,我会继续支持!
- 2023-03-20 01:29:10
-
- 结实的冬瓜
- 这篇文章太及时了,很详细,很棒,mark,关注老哥了!希望老哥能多写Golang相关的文章。
- 2023-03-17 01:19:56
-
- 阳光的康乃馨
- 真优秀,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢师傅分享文章内容!
- 2023-01-09 03:48:18
-
- Golang · Go教程 | 39秒前 |
- Golang环境搭建与数据库配置教程
- 165浏览 收藏
-
- Golang · Go教程 | 21分钟前 |
- GeanyGo语言编辑技巧分享
- 263浏览 收藏
-
- Golang · Go教程 | 22分钟前 |
- Golang反射缺陷及替代方案解析
- 299浏览 收藏
-
- Golang · Go教程 | 25分钟前 |
- GolangServerless冷启动优化技巧
- 238浏览 收藏
-
- Golang · Go教程 | 31分钟前 |
- GolangTCP粘包处理与自定义协议详解
- 420浏览 收藏
-
- Golang · Go教程 | 33分钟前 | golang net/http 标准库 CRUD RESTfulAPI
- Golang标准库搭建RESTfulAPI教程
- 346浏览 收藏
-
- Golang · Go教程 | 35分钟前 |
- 优化GolangHTTPKeep-Alive设置提升性能
- 377浏览 收藏
-
- Golang · Go教程 | 38分钟前 |
- Golangcontext如何影响Web开发?
- 385浏览 收藏
-
- Golang · Go教程 | 39分钟前 |
- Golang跨语言绑定:cgo与SWIG使用教程
- 418浏览 收藏
-
- Golang · Go教程 | 44分钟前 |
- Golangio库流式操作:Reader与Writer详解
- 141浏览 收藏
-
- Golang · Go教程 | 50分钟前 |
- Golang降低GC停顿技巧:手动内存管理方法
- 162浏览 收藏
-
- Golang · Go教程 | 56分钟前 |
- Golang构建BFF服务,定制客户端后端方案
- 170浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 千音漫语
- 千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
- 96次使用
-
- MiniWork
- MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
- 89次使用
-
- NoCode
- NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
- 107次使用
-
- 达医智影
- 达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
- 98次使用
-
- 智慧芽Eureka
- 智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
- 99次使用
-
- Golang中的错误处理的示例详解
- 2022-12-22 119浏览
-
- Go中的错误和异常处理最佳实践方法
- 2023-01-09 100浏览
-
- Go语言异常处理(Panic和recovering)用法详解
- 2023-01-07 197浏览
-
- Go 处理大数组使用 for range 和 for 循环的区别
- 2023-01-09 272浏览
-
- golang为什么要统一错误处理
- 2022-12-29 172浏览