位置: IT常识 - 正文

golang中的错误处理(golang eventbus)

编辑:rootadmin
0.1、索引 https://waterflow.link/articles/1666716727236 1、panic 当我们执行panic的时候会结束下面的流程: package main import "fmt" func main() { fmt.Println("hello") panic ... 0.1、索引

推荐整理分享golang中的错误处理(golang eventbus),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:go语言错误处理,golang errors,golang queue,golang errors.new,golang异常处理最佳实践,golang异常处理最佳实践,golang 错误处理,golang 错误处理,内容如对您有帮助,希望把文章链接给更多的朋友!

https://waterflow.link/articles/1666716727236

1、panic

当我们执行panic的时候会结束下面的流程:

package mainimport "fmt"func main() {fmt.Println("hello")panic("stop")fmt.Println("world")}go run 9.gohellopanic: stop

但是panic也是可以捕获的,我们可以使用defer和recover实现:

package mainimport "fmt"func main() {defer func() {if r := recover(); r != nil {fmt.Println("recover: ", r)}}()fmt.Println("hello")panic("stop")fmt.Println("world")}go run 9.gohellorecover: stop

那什么时候适合panic呢?在 Go 中,panic 用于表示真正的异常,例如程序错误。我们经常会在一些内置包里面看到panic的身影。

比如strings.Repeat重复返回一个由字符串 s 的计数副本组成的新字符串:

func Repeat(s string, count int) string {if count == 0 {return ""}// if count < 0 {panic("strings: negative Repeat count")} else if len(s)*count/count != len(s) {panic("strings: Repeat count causes overflow")}...}

我们可以看到当重复的次数小于0或者重复count次之后s的长度溢出,程序会直接panic,而不是返回错误。这时因为strings包限制了error的使用,所以在程序错误时会直接panic。

还有一个例子是关于正则表达式的例子:

package mainimport ("fmt""regexp")func main() {pattern := "a[a-z]b*" // 1compile, err := regexp.Compile(pattern) // 2if err != nil { // 2fmt.Println("compile err: ", err)return} // 3allString := compile.FindAllString("acbcdadb", 3)fmt.Println(allString)}编写一个正则表达式调用Compile,解析正则表达式,如果成功,返回用于匹配文本的 Regexp 对象。否则返回错误利用正则,在输入的字符串中,获取所有的匹配字符

可以看到如果上面正则解析失败是可以继续往下执行的,但是regexp包中还有另外一个方法MustCompile:

func MustCompile(str string) *Regexp {regexp, err := Compile(str)if err != nil {panic(`regexp: Compile(` + quote(str) + `): ` + err.Error())}return regexp}

这个方法说明正则的解析是强依赖的,如果解析错误,直接panic结束程序。用户可以根据实际情况选择。

但是实际开发中我们还是要谨慎使用panic,因为它会使程序结束运行(除非我们调用defer recover)

2、包装错误golang中的错误处理(golang eventbus)

错误包装是将错误包装或者打包在一个包装容器中,这样的话我们就可以追溯到源错误。错误包装的主要作用就是:

为错误添加上下文将错误标记为特定类型的错误

我们可以看一个访问数据库的例子:

package mainimport ("fmt""github.com/pkg/errors")type Courseware struct {Id int64Code stringName string}func getCourseware(id int64) (*Courseware, error) {courseware, err := getFromDB(id)if err != nil {return nil, errors.Wrap(err, "六月的想访问这个课件") // 2}return courseware, nil}func getFromDB(id int64) (*Courseware, error) {return nil, errors.New("permission denied") // 1}func main() {_, err := getCourseware(11)if err != nil {fmt.Println(err)}}访问数据库时我们返回了原始的错误信息到上层我们添加了一些自定义的上下文信息go run 9.go六月的想访问这个课件: permission denied

当然我们也可以将错误包装成我们自定义类型的错误,我们稍微修改下上面的例子:

package mainimport ("fmt""github.com/pkg/errors")type Courseware struct {Id int64Code stringName string}// 1type ForbiddenError struct {Err error}// 2func (e *ForbiddenError) Error() string {return "Forbidden: " + e.Err.Error()}func getCourseware(id int64) (*Courseware, error) {courseware, err := getFromDB(id)if err != nil {return nil, &ForbiddenError{err} // 4}return courseware, nil}func getFromDB(id int64) (*Courseware, error) {return nil, errors.New("permission denied") // 3}func main() {_, err := getCourseware(11)if err != nil {fmt.Println(err)}}首先我们自定义了ForbiddenError的错误类型我们实现了error接口访问数据库抛出原始错误上层返回ForbiddenError类型的错误go run 9.goForbidden: permission denied

当然我们也可以不用创建自定义错误的类型,去包装错误添加上下文:

package mainimport ("fmt""github.com/pkg/errors")type Courseware struct {Id int64Code stringName string}func getCourseware(id int64) (*Courseware, error) {courseware, err := getFromDB(id)if err != nil {return nil, fmt.Errorf("another wrap err: %w", err) // 1}return courseware, nil}func getFromDB(id int64) (*Courseware, error) {return nil, errors.New("permission denied")}func main() {_, err := getCourseware(11)if err != nil {fmt.Println(err)}}使用%w包装错误

使用这的好处是我们可以追溯到源错误,从而方便我们做一些特殊的处理。

还有一种方式是使用:

return nil, fmt.Errorf("another wrap err: %v", err)

%v的方式不会包装错误,所以无法追溯到源错误,但往往有时候我们会选择这种方式,而不用%w的方式。%w的方式虽然能包装源错误,但往往我们会通过源错误去做一些处理,假如源错误被修改,那包装这个源错误的相关错误都需要做响应变化。

3、错误类型判断

我们扩展一下上面查询课件的例子。现在我们有这样的判断,如果传进来的id不合法我们返回400错误,如果查询数据库报错我们返回500错误,我们可以像下面这样写:

package mainimport ("fmt""github.com/pkg/errors")type Courseware struct {Id int64Code stringName string}type ForbiddenError struct {Err error}func (e *ForbiddenError) Error() string {return "Forbidden: " + e.Err.Error()}func getCourseware(id int64) (*Courseware, error) {if id <= 0 {return nil, fmt.Errorf("invalid id: %d", id)}courseware, err := getFromDB(id)if err != nil {return nil, &ForbiddenError{err}}return courseware, nil}func getFromDB(id int64) (*Courseware, error) {return nil, errors.New("permission denied")}func main() {_, err := getCourseware(500) // 我们可以修改这里的id看下打印的结构if err != nil {switch err := err.(type) {case *ForbiddenError:fmt.Println("500 err: ", err)default:fmt.Println("400 err: ", err)}}}go run 9.go500 err: Forbidden: permission denied

这样看起来好像也没什么问题,现在我们稍微修改下代码,把上面ForbiddenError包装一下:

package mainimport ("fmt""github.com/pkg/errors")type Courseware struct {Id int64Code stringName string}type ForbiddenError struct {Err error}func (e *ForbiddenError) Error() string {return "Forbidden: " + e.Err.Error()}func getCourseware(id int64) (*Courseware, error) {if id <= 0 {return nil, fmt.Errorf("invalid id: %d", id)}courseware, err := getFromDB(id)if err != nil {return nil, fmt.Errorf("wrap err: %w", &ForbiddenError{err}) // 这里包装了一层错误}return courseware, nil}func getFromDB(id int64) (*Courseware, error) {return nil, errors.New("permission denied")}func main() {_, err := getCourseware(500)if err != nil {switch err := err.(type) {case *ForbiddenError:fmt.Println("500 err: ", err)default:fmt.Println("400 err: ", err)}}}go run 9.go400 err: wrap err: Forbidden: permission denied

可以看到我们的Forbidden错误进到了400里面,这并不是我们想要的结果。之所以会这样,是因为在ForbiddenError的外面又包装了一层Error错误,使用类型断言的时候判断出来的是Error错误,所以进到了400分支。

这里我们可以使用errors.As方法,它会递归调用Unwrap方法,找到错误链中第一个与target匹配的方法:

package mainimport ("fmt""github.com/pkg/errors")type Courseware struct {Id int64Code stringName string}type ForbiddenError struct {Err error}func (e *ForbiddenError) Error() string {return "Forbidden: " + e.Err.Error()}func getCourseware(id int64) (*Courseware, error) {if id <= 0 {return nil, fmt.Errorf("invalid id: %d", id)}courseware, err := getFromDB(id)if err != nil {return nil, fmt.Errorf("wrap err: %w", &ForbiddenError{err})}return courseware, nil}func getFromDB(id int64) (*Courseware, error) {return nil, errors.New("permission denied")}func main() {_, err := getCourseware(500)if err != nil {var f *ForbiddenError // 这里实现了*ForbiddenError接口,不然会panicif errors.As(err, &f) { // 找到匹配的错误fmt.Println("500 err: ", err)} else {fmt.Println("400 err: ", err)}}}go run 9.go500 err: wrap err: Forbidden: permission denied4、错误值判断

在代码中或者mysql库或者io库中我们经常会看到这样的全局错误:

var ErrCourseware = errors.New("courseware")

这种错误我们称之为哨兵错误。一般数据库没查到ErrNoRows或者io读到了EOF错误,这些特定的错误可以帮助我们做一些特殊的处理。

一般我们会直接用==号判断错误值,但是就像上面的如果错误被包装哪我们就不好去判断了。好在errors包中提供了errors.Is方法,通过递归调用Unwrap判断错误链中是否与目标错误相匹配的错误值:

if err != nil { if errors.Is(err, ErrCourseware) { // ... } else { // ... }}
本文链接地址:https://www.jiuchutong.com/zhishi/310534.html 转载请保留说明!

上一篇:python变量中self的添加(Python变量中forain)

下一篇:Python之字典遍历元素(4种方式)(python字典遍历key)

  • 戴尔相机在哪里打开(戴尔拍照快捷键)

    戴尔相机在哪里打开(戴尔拍照快捷键)

  • 怎么注销支付宝(怎么注销支付宝基金账户)

    怎么注销支付宝(怎么注销支付宝基金账户)

  • 苹果授权店退换货政策是(苹果授权店退换货政策)

    苹果授权店退换货政策是(苹果授权店退换货政策)

  • 待付款怎么取消(待付款怎么取消订单)

    待付款怎么取消(待付款怎么取消订单)

  • vb中val什么意思(vb中val函数的使用方法)

    vb中val什么意思(vb中val函数的使用方法)

  • zoom有回放吗(zoom的回放怎么看)

    zoom有回放吗(zoom的回放怎么看)

  • 跳转到原网页什么意思(网页跳转到其他网页代码)

    跳转到原网页什么意思(网页跳转到其他网页代码)

  • 视频聊天流量耗的多吗(视频聊天 流量)

    视频聊天流量耗的多吗(视频聊天 流量)

  • nice有照片数量但不显示(nice怎么设置照片权限)

    nice有照片数量但不显示(nice怎么设置照片权限)

  • 小米笔记本numlock键在哪(小米笔记本numlock键)

    小米笔记本numlock键在哪(小米笔记本numlock键)

  • 前置摄像头是什么意思(前置摄像头是什么时候出来的)

    前置摄像头是什么意思(前置摄像头是什么时候出来的)

  • 段落之间空白无法删除(段落之间没有空格怎么弄)

    段落之间空白无法删除(段落之间没有空格怎么弄)

  • 主板黄灯亮无法开机(主板亮黄灯开机没反应)

    主板黄灯亮无法开机(主板亮黄灯开机没反应)

  • 芒果tv怎么绑定微信(芒果TV怎么绑定电视)

    芒果tv怎么绑定微信(芒果TV怎么绑定电视)

  • 程序和数据在计算机中用什么表示(程序和数据在计算机中以什么形式存储)

    程序和数据在计算机中用什么表示(程序和数据在计算机中以什么形式存储)

  • 写字板可以干什么(写字板可以用来画画吗)

    写字板可以干什么(写字板可以用来画画吗)

  • 苹果11为什么没有电池百分比(苹果11为什么没有专注模式)

    苹果11为什么没有电池百分比(苹果11为什么没有专注模式)

  • 在word中h这个toc域开关有何作用(在word当中)

    在word中h这个toc域开关有何作用(在word当中)

  • 华硕主板360和365区别(华硕365主板配置清单)

    华硕主板360和365区别(华硕365主板配置清单)

  • 手机装了内存卡怎么装两个手机卡(手机装了内存卡怎么看内存)

    手机装了内存卡怎么装两个手机卡(手机装了内存卡怎么看内存)

  • 手机微信为什么老掉线(手机微信为什么发不了语音)

    手机微信为什么老掉线(手机微信为什么发不了语音)

  • 88vip怎么取消续费(88vip怎么取消续订)

    88vip怎么取消续费(88vip怎么取消续订)

  • 华为荣耀7怎么装双卡(华为荣耀7怎么装卡)

    华为荣耀7怎么装双卡(华为荣耀7怎么装卡)

  • 电脑源坏了有什么症状(电脑电源坏的几种现象)

    电脑源坏了有什么症状(电脑电源坏的几种现象)

  • element-ui table使用type=‘selection‘复选框全禁用-全选禁用

    element-ui table使用type=‘selection‘复选框全禁用-全选禁用

  • 未交增值税借方表示什么意思
  • 全国税务师考试报名时间
  • 小规模租金可以按1%
  • 生产车间维修费是制造费用还是管理费用
  • 企业注销增值税留抵可以退吗
  • 应交税费明细分类账
  • 个体工商户营业执照年检
  • 五证一户什么意思
  • 金三打印客户端插件如何下载?
  • 待解报预算收入扣款怎么做账
  • 拿提成的工资怎么算
  • 事业单位固定资产处置流程
  • 营改增后兼营非应税劳务的税务处理怎么做?
  • 每季度不超过9万
  • 财政拨款收入如何记账
  • wifi上传速度为0的解决方法
  • linux 列表
  • 苹果iphone6s plus打电话没有人名
  • mac辅助功能键盘
  • wifan
  • 母公司向子公司注入资产
  • 发票是自己买的吗
  • 不良修复体的危害
  • 企业所得额不能扣除
  • php的fopen
  • axios提交文件
  • 前端这点事
  • php遍历目录
  • 承兑贴现会计分录怎么做
  • kaldi官网
  • 变量与数据
  • linux 运行php
  • 金蝶销售明细查不出来
  • 套期工具会计核算
  • 电子税务局网开电子发票
  • 长期待摊费用如何做分录
  • 外出经营税收缴纳
  • 公司注销后实收资本要交税吗
  • 工会经费按什么比例返还
  • 未计提坏账准备金额在哪个表
  • 报关单金额大于出口发票金额
  • 湖南残保金优惠政策
  • 公司客户招待费用标准
  • 支付技术转让费怎么入账
  • 怎样可以冲销企业微信
  • 科目汇总表一张几行啊
  • 承租人和共同承租人
  • 交易性金融资产公允价值变动计入
  • 出纳怎么做账
  • mysql数据库优化总结
  • linux系统怎样安装
  • Solaris 10 Openssh安装和配置
  • ubuntu鼠标不能移动
  • liunx系统常用命令
  • cygwin下载教程
  • Linux系统root密码忘记怎么重置
  • ssh实现登录功能
  • win8 重置
  • win10允许访问
  • win7系统怎么设置不休眠
  • 怎么看win8.1的版本
  • efs加密解除
  • win7系统桌面不能新建文件夹
  • windows8安装步骤
  • win10玩饥荒
  • win7打开软件总提示错误怎么解决
  • Win10 Mobile 10586.164中文版升级截图曝光
  • html用法
  • unity3D游戏开发
  • 女鬼照片
  • android图形架构
  • 定时软件有哪些
  • angular js表达式
  • jquery.validate提示错误信息位置方法
  • python多线程菜鸟教程
  • jQuery+Ajax实现限制查询间隔的方法
  • 医保断交一年怎么续交
  • 深圳监狱在哪个位置
  • 税务报道可以学什么专业
  • 农夫山泉水质量合格吗
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设