Go语言的错误和异常处理:error、panic和recover

news2024/11/25 8:17:23

目录

【error类型】

error的基本用法

error.Is 用法

封装自定义错误结构体

error.As 用法

错误行为特征检视策略

【异常panic和recover】

panic

recover

 panic 和 os.Exit

如何正确应对panic


【error类型】

error的基本用法

在Go语言中,一般使用 error 这个接口类型表示错误,并且通常将 error 类型返回值放在返回值列表的末尾,比如下面这样:

package main

import (
	"errors"
	"fmt"
	"time"
)

func main() {
    if caclResult, err := cacl(-1, 2); err == nil {
		fmt.Println(caclResult)
	} else {
		fmt.Println(err)
	}
}

func cacl(a int, b int) (int, error) {
	if a < 0 || b < 0 {
		return -1, errors.New("a或者b不能小于0")
	}
	return a + b, nil
}

error 接口类型是 Go 原生内置的类型,它的定义如下:

// $GOROOT/src/builtin/builtin.go
    type interface error {
    Error() string
}

Go 在标准库中提供了两种构造错误值的方法: errors.New 和 fmt.Errorf,这两种方法只限于以字符串形式返回错误信息。

err := errors.New("your first demo error")
errWithCtx = fmt.Errorf("index %d is out of bounds", i)

也可以使用 error 类型来处理错误,比如标准库中的 net 包就定义了包含错误上下文的错误类型:

// $GOROOT/src/net/net.go
type OpError struct {
    Op string
    Net string
    Source Addr
    Addr Addr
    Err error //包含错误上下文的错误类型
}

一般在函数中不关心返回错误值携带的具体上下文信息,只要发生错误就进入唯一的错误处理执行路径,比如下面这段代码:

package main

import (
	"errors"
	"fmt"
)

func main() {
	fmt.Println(getDbInfo())
}

func getDbInfo() (string, error) {
	err := checkDbConnect()
	if err != nil { //此处不关心返回错误值携带的具体上下文信息,只要发生错误就返回
		return "", err
	}
	return "mysql", nil
}

func checkDbConnect() error {
	foo := false
	if !foo {
		return errors.New("数据库连接失败")
	}
	return nil
}

也可以判断具体的错误类型后再处理相关业务逻辑:

package main

import (
	"errors"
	"fmt"
)

var (
	ErrDbConnectFailed = errors.New("数据库连接失败")
	ErrDbUsernameError = errors.New("数据库用户名错误")
	ErrDbPasswordError = errors.New("数据库密码错误")
    ErrCodeTest        = fmt.Errorf("code:%d", 200)
)

func main() {
    fmt.Println(ErrCodeTest)               //code:200
	fmt.Printf("%T\n", ErrCodeTest)        //*errors.errorString
	fmt.Printf("%T\n", ErrDbConnectFailed) //*errors.errorString
	fmt.Println(getDbInfo())
	updateDb()
}

// 判断具体的错误类型再处理相关逻辑
func updateDb() {
	err := checkDbConnect()
	if err != nil {
		switch err {
		case ErrDbConnectFailed:
			fmt.Println("处理数据库连接失败的逻辑...")
			return
		case ErrDbUsernameError:
			fmt.Println("处理数据库用户名错误的逻辑...")
			return
		case ErrDbPasswordError:
			fmt.Println("处理数据库密码错误的逻辑...")
			return
		}
	}
	fmt.Println("操作处理完成")
}

func checkDbConnect() error {
	foo := false
	if !foo {
		return ErrDbConnectFailed
	}
	return nil
}

再比如,校验年龄是否合法:

package main

import (
	"errors"
	"fmt"
)

func main() {
    if err := checkAge(-1); err != nil {
		fmt.Println(err) //年龄是:-1,不合法
		return
	}
    fmt.Println("这里不会执行")
}

func checkAge(age int) error {
	if age < 0 {
		//return errors.New("年龄不合法") //返回error对象
		return fmt.Errorf("年龄是:%d,不合法", age)
	}
	fmt.Println("年龄是:", age)
	return nil
}

error.Is 用法

上面代码中错误信息耦合到了业务中,不利于维护。从 Go 1.13 版本开始,标准库 errors 包提供了 Is 函数 用于错误处理方对错误值的检视,上面的代码可以改造如下:

func updateDbV2() {
	err := checkDbConnect()
	if errors.Is(err, ErrDbConnectFailed) {
		fmt.Println("处理数据库连接失败的逻辑...")
		return
	}
	if errors.Is(err, ErrDbUsernameError) {
		fmt.Println("处理数据库用户名错误的逻辑...")
		return
	}
	if errors.Is(err, ErrDbPasswordError) {
		fmt.Println("处理数据库密码错误的逻辑...")
		return
	}
	fmt.Println("操作处理完成")
}

使用 errors.Is 可以根据错误链找到最底层的错误信息,并判断它们是否是同一类。再看下面的代码:

func errorIsDemo() {
	//使用 fmt.Errorf 对错误变量包装
	err1 := fmt.Errorf("err1: %w", ErrDbConnectFailed) //基于 ErrDbConnectFailed 包装出 err1
	err2 := fmt.Errorf("err2: %w", err1)               //基于 err1 包装出 err2
	fmt.Println(err1)                                  //输出: err1: 数据库连接失败
	fmt.Println(err2)                                  //输出: err2: err1: 数据库连接失败
	fmt.Println(err2 == ErrDbConnectFailed)            //false
	fmt.Println(errors.Is(err1, ErrDbConnectFailed))   //true
	fmt.Println(errors.Is(err2, ErrDbConnectFailed))   //true
	fmt.Println(errors.Is(err2, err1))                 //true
	fmt.Println(errors.Is(err1, err2))                 //false

    //可以使用 errors.Unwrap 获取被嵌套的 error
	fmt.Println(errors.Unwrap(err1)) //数据库连接失败
	fmt.Println(errors.Unwrap(err2)) //err1: 数据库连接失败
}

因此, 在 Go 1.13 及后续版本中建议使用 errors.Is 方法去检视某个错误值是否属于某个预期的错误值,而尽量避免使用 if/switch 判断。

封装自定义错误结构体

如果遇到错误的地方需要提供更多“上下文”信息,可以自定义一个错误结构体,一般以 XXXError形式命名,比如标准库里面  json 包的一个方法的实现:

// $GOROOT/src/encoding/json/decode.go
type UnmarshalTypeError struct {
    Value string
    Type reflect.Type
    Offset int64
    Struct string
    Field string
}

// $GOROOT/src/encoding/json/decode.go
func (d *decodeState) addErrorContext(err error) error {
	if d.errorContext.Struct != nil || len(d.errorContext.FieldStack) > 0 {
		switch err := err.(type) {
		case *UnmarshalTypeError:
			err.Struct = d.errorContext.Struct.Name()
			err.Field = strings.Join(d.errorContext.FieldStack, ".")
			return err
		}
	}
	return err
}

error.As 用法

从 Go 1.13 版本开始,标准库 errors 包提供了As函数给错误处理方检视错误值。As函数类似于通过类型断言判断一个 error 类型变量是否为其它自定义错误类型。

// 类似 if e, ok := err.(*MyError); ok { … }
var e *MyError
if errors.As(err, &e) {
    // 如果err类型为 *MyError,变量e将被设置为对应的错误值
}

和 erros.Is 类似,errors.As 也可以根据错误链找到最底层的错误信息。

// 自定义一个error结构体
type MyError struct {
	errCode int    //错误码
	errMsg  string //错误信息
}

func (e *MyError) Error() string {
	return fmt.Sprintf("错误码:%d,错误信息:%v", e.errCode, e.errMsg)
}

func errorAsDemo() {
	var err = &MyError{errCode: 500, errMsg: "服务器异常"}
	err1 := fmt.Errorf("err1: %w", err)
	err2 := fmt.Errorf("err2: %w", err1)
	fmt.Println(err, " -- ", err1, "--", err2) //错误码:500,错误信息:服务器异常  --  err1: 错误码:500,错误信息:服务器异常 -- err2: err1: 错误码:500,错误信息:服务器异常

	var e *MyError
	fmt.Println(errors.As(err2, &e))    //true
	fmt.Println(errors.As(err1, &e))    //true
	fmt.Println(errors.As(err2, &err1)) //true
	fmt.Println(errors.As(err1, &err2)) //true
}

同样的,如果使用的是 Go 1.13 及后续版本,就尽量使用 errors.As 方法去检视某个错误值是否是某自定义错误类型的实例。

错误行为特征检视策略

将某个包中的错误类型归类,统一提取出一些公共的错误行为特征,并将这些错误行为特征放入一个公开的接口类型中。这种方式也被叫做错误行为特征检视策略。比如标准库 net.Error 接口就口包含两个用于判断错误行为特征的方法,错误处理方只需要依赖这个公共接口就可以检视具体错误值的错误行为特征信息,并根据这些信息做出后续错误处理分支选择的决策。

// $GOROOT/src/net/net.go
type Error interface {
    error
    Timeout() bool //用来判断是否是超时错误
    Temporary() bool //用于判断是否是临时错误
}

为了保证函数的健壮性,需要注意下面几个原则: 

  • 不要相信任何外部输入的参数:函数需要对所有输入的参数进行合法性的检查。一旦发现问题,立即终止函数的执行,返回预设的错误值。
  • 不要忽略任何一个错误:调用标准库或第三方包的函数或方法时不能假定它一定会成功,需要显式地检查这些调用返回的错误值,一旦发现错误就要及时终止函数执行,防止错误继续传播。
  • 不要认为异常不会发生:异常不是错误,错误是可预期的,但异常却是少见的、意料之外的,比如除以0;虽然少见,但不能认为异常不会发生,所以需要关注对异常的捕捉和恢复。

【异常panic和recover】

panic

在PHP或者Java程序中,异常一般使用 try...catch...finally 表示,而在 Go 语言中用 panic 表示异常,但又和PHP的 try...catch...finally 不完全一样。在 Go 中 panic 主要有两类来源,一类是来自 Go 运行时,另一类则是 Go 开发人员通过 panic 函数主动触发的。panic可以接收一个 interface{} 类型的参数,也就是任意类型。如果异常出现了但没有被捕获并恢复,就会终止程序的执行,即便出现异常的位置不在主Goroutine 中也会这样。

package main

func main() {
	println("main start")
	user()
	println("main end")
}

func user() {
	println("user start")
	goods()
	println("user end")
}
func goods() {
	println("goods start")
	panic("在 goods 中触发了 panic")
	info()               //Unreachable code
	println("goods end") //Unreachable code
}
func info() {
	println("info start")
	println("info end")
}

可以看到,函数的调用次序为 main -> user -> goods -> info, 在 goods 函数中手动触发 panic,不会再往后执行,不仅 info 没有任何执行,并且 main、user、goods 的 end 部分也都没有执行。无论在哪个 Goroutine 中发生未被恢复的 panic 整个程序都将崩溃退出。

recover

Go 中可以通过 recover 函数捕捉 panic 并恢复程序正常执行,recover 是 Go 内置的专门用于恢复 panic 的函数,它必须被放在一个 defer 函数中才能生效。如果 recover 捕捉到 panic,它就会返回以 panic 的具体内容为错误上下文信息的错误值。如果没有 panic 发生,那么 recover 将返回 nil。

修改上面的测试代码如下:

func goods() {
	defer func() {
		if e := recover(); e != nil {
			fmt.Println("recover the panic:", e)
		}
	}()

	println("goods start")
	panic(errors.New("在 goods 中触发了 panic"))
	info()               //Unreachable code
	println("goods end") //Unreachable code
}

修改之后,main、user 的 end  部分可以执行了。 因此如果作为 API 函数的作者,一定不要将 panic 当作错误返回给 API 调用者。 

 panic 和 os.Exit

• panic ⽤于不可以恢复的错误
• panic 退出前会执⾏ defer 指定的内容

• os.Exit 退出时不会调⽤ defer 指定的函数
• os.Exit 退出时不输出当前调⽤栈信息

现在把上面的panic改成os.Exit试试:

func goods() {
	defer func() {
		if e := recover(); e != nil {
			fmt.Println("recover the panic:", e)
		}
	}()

	println("goods start")
	//panic(errors.New("在 goods 中触发了 panic"))
	os.Exit(-1)
	info()               //Unreachable code
	println("goods end") //Unreachable code
}

如何正确应对panic

并不是在所有场景下都应该使用panic和recover,因为很多函数非常简单,根本不会出现 panic 情况,如果增加了 panic 捕获和恢复反倒会增加函数的复杂性,也会增加开发的实现过程。而且在 Go语言的函数和defer用法_浮尘笔记的博客-CSDN博客 中也说过,带有 defer 的函数执行开销会比不带 defer 的函数的执行开销大一些。所以应该从以下几个方面考虑使用panic的场景:

(1)针对各种应用对 panic 忍受度的差异,采取的应对策略也应该不同。

比如后端 HTTP 服务器这样的关键系统就需要在特定位置捕捉并恢复 panic 以保证服务器整体的健壮度,Go 标准库中的 http server 就是这样设计的。它采用的是每个客户端连接都使用一个单独的 Goroutine 进行处理并发模型,客户端一旦与 http server 连接成功,http server 就会为这个连接新创建一个 Goroutine,并在这 Goroutine 中执行对应连接的 serve 方法来处理这条连接上的客户端请求。为了保证处理某一个客户端连接的 Goroutine 出现 panic 时不影响到 http server 主 Goroutine 的运行,Go 标准库在 serve 方法中加入了对 panic 的捕捉与恢复。

下面是 serve 方法的部分代码片段:

// $GOROOT/src/net/http/server.go

func (c *conn) serve(ctx context.Context) {
	c.remoteAddr = c.rwc.RemoteAddr().String()
	ctx = context.WithValue(ctx, LocalAddrContextKey, c.rwc.LocalAddr())
    // serve 方法在一开始处就设置了 defer 函数,并在该函数中捕捉并恢复了可能出现的 panic。
    // 这样即便处理某个客户端连接的 Goroutine 出现 panic,处理其他连接 Goroutine 以及 http server 自身都不会受到影响
	defer func() {
		if err := recover(); err != nil && err != ErrAbortHandler {
			//...
		}
		if !c.hijacked() {
			//...
		}
	}()
}

(2)使用panic提示潜在 bug

Go 语言标准库中没有提供断言之类的辅助函数,但可以使用 panic 模拟断言对潜在 bug 的提示功能。在 Go 标准库中,大多数panic 的使用都是充当类似断言的作用的。

比如下面就是标准库 encoding/json 包使用 panic 指示潜在 bug 的一个例子:

// $GOROOT/src/encoding/json/decode.go

// 当一些本不该发生的事情导致我们结束处理时,phasePanicMsg将被用作panic消息
// 它可以指示JSON解码器中的bug,或者在解码器执行时还有其他代码正在修改数据切片。
const phasePanicMsg = "JSON decoder out of sync - data changing underfoot?"

func (d *decodeState) valueQuoted() interface{} {
	switch d.opcode {
	default: //如果程序执行流进入了 default 分支就会引发 panic,用来提示开发人员:这里很可能是一个 bug
		panic(phasePanicMsg)
	case scanBeginArray, scanBeginObject:
		//...
	case scanBeginLiteral:
		//...
	}
	return unquotedValue{}
}

再比如, json 包的 encode.go 中也有使用 panic 做潜在 bug 提示的例子:

// $GOROOT/src/encoding/json/encode.go
func (w *reflectWithString) resolve() error {
	switch w.k.Kind() {
	case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int6
		w.ks = strconv.FormatInt(w.k.Int(), 10)
		return nil
	case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.
		w.ks = strconv.FormatUint(w.k.Uint(), 10)
		return nil
	}
	panic("unexpected map key type") //这行代码就相当于一个“代码逻辑不会走到这里”的断言,一旦触发就表示很可能是一个 bug
}

最后再补充一点,有些时候不要对 panic 进行 recover,因为出现了panic就说明程序存在问题,需要解决。具体如果应对也应该视具体情况而定。

源代码:https://gitee.com/rxbook/go-demo-2023/tree/master/basic/go02

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/534682.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【P1003 [NOIP2011 提高组] 铺地毯】

[NOIP2011 提高组] 铺地毯 题目描述 为了准备一个独特的颁奖典礼&#xff0c;组织者在会场的一片矩形区域&#xff08;可看做是平面直角坐标系的第一象限&#xff09;铺上一些矩形地毯。一共有 n n n 张地毯&#xff0c;编号从 1 1 1 到 n n n。现在将这些地毯按照编号从小…

Redis单线程 Vs 多线程

Redis单线程 Vs 多线程 一 面试题引入1.1 Redis到底是单线程还是多线程&#xff1f;1.2 IO多路复用1.3 Redis为什么快&#xff1f;1.4 Subtopic 二 Redis为什么选择单线程&#xff1f;2.1 这种问法其实并不严谨&#xff0c;为啥这么说&#xff1f;2.2 Redis是单线程究竟何意&am…

什么是JavaScript?为什么需要学习它?

JavaScript是一种广泛使用的编程语言&#xff0c;它被用于开发Web应用程序、桌面应用程序和移动应用程序。它的出现可以追溯到1995年&#xff0c;由瑞典计算机科学家Tim Bergling和美国计算机科学家John Resig共同开发。 JavaScript的历史可以追溯到20世纪90年代&#xff0c;当…

完美解决:由于找不到MSVR100.dll ,无法继续执行代码

当我们在运行某一个软件时&#xff0c;突然提示找不到MSVCR100.dll&#xff0c;我相信有不少用户都遇到过这种情况&#xff0c;并且在重新安装软件后还是无法解决。那么电脑提示找不到MSVCR100.dll该怎办呢? MSVCR100.dll是什么&#xff1f; 在解决找不到MSVCR100.dll这个问…

RabbitMQ之交换机详解

1 Exchages ​ 我们假设的是工作队列背后&#xff0c;每个任务都恰好交付给一个消费者(工作进程)。在这一部分中&#xff0c;我们将做一些完全不同的事情&#xff0c;我们将消息传达给多个消费者。这种模式 称为 ”发布/订阅“。 ​ 为了说明这种模式&#xff0c;我们将构建一…

vi和vim编辑器介绍与使用

VI 和 VIM 编辑器是 Unix 和 Linux 操作系统中最常用的文本编辑工具之一。虽然它们都用于编辑文本文件&#xff0c;但它们有一些不同之处。本文将对 VI 和 VIM 编辑器进行介绍&#xff0c;帮助你更好地了解编辑器的特性和优点。 Linux下常见的文本编辑器有&#xff1a; emacsp…

Unity解决在摄像机上面设置了TargetTexture后获取屏幕坐标不准的问题

大家好&#xff0c;我是阿赵 这里来分享一个最近遇到的小问题。 一、发现问题 如果我们想将3D模型放在UI上&#xff0c;一个比较普遍的做法是&#xff1a; 用一个单独的摄像机&#xff0c;把3D模型拍下来&#xff0c;并转成RenderTexture&#xff0c;贴到RawImage上。 那么如…

枚举类型enum详解

概述 enum是C语言中的一个关键字&#xff0c;enum叫枚举数据类型&#xff0c;枚举数据类型描述的是一组整型值的集合&#xff08;这句话其实不太妥当&#xff09;&#xff0c;因为枚举类型是一种基本数据类型&#xff0c;而不是一种构造类型&#xff0c;它不能再分解成什么基本…

架构设计如何绘图?

大家好&#xff0c;我是易安&#xff01; 很多同学技术能力很强&#xff0c;架构设计也做得很好&#xff0c;但是在给别人讲解的时候&#xff0c;总感觉像是“茶壶里煮饺子&#xff0c;有货倒不出”。 其实&#xff0c;在为新员工培训系统架构、给领导汇报技术规划、上技术大会…

详解c++STL—容器vector

一、vector基本概念 功能&#xff1a; vector数据结构和数组非常相似&#xff0c;也称为单端数组 vector与普通数组的区别&#xff1a; 不同之处在于数组是静态空间&#xff0c;而vector可以动态扩展 动态扩展&#xff1a; 并不是在原空间之后续接新空间&#xff0c;而是找…

阿里云ECS服务器实例挂载数据盘步骤

1、首先登录阿里云ECS服务器&#xff0c;通过 df -h 命令查看当前磁盘挂载情况&#xff0c;可以发现只有系统盘40G 2、通过 fdisk -l 命令查看磁盘情况&#xff0c;可以发现有两个盘&#xff1a; 系统盘 /dev/vda: 42.9GB&#xff0c; 数据盘 /dev/vdb: 322.1GB 3、运行 fdisk…

c++进阶项目—基于多态的职工管理系统

一、管理系统需求 公司中职工分为三类&#xff1a;普通员工、经理、老板 显示信息时&#xff0c;需要显示职工编号、职工姓名、职工岗位、以及职责 1、普通员工职责&#xff1a;完成经理交给的任务 2、经理职责&#xff1a;完成老板交给的任务&#xff0c;并下发任务给员工…

“Shell“正则表达式;文本三剑客Sed

文章目录 一.正则表达式1.1正则表达式介绍1.2命令演示1.3正则表达式实验演示 二.Sed编辑器2.1Sed介绍2.2操作命令演示2.3替换2.4插入2.4总结&#xff1a; 一.正则表达式 1.1正则表达式介绍 通常用于判断语句中&#xff0c;用来检查某一字符串是否满足某一格式 正则表达式是由…

unity,如何让人物随着鼠标滑动而转向?

介绍 unity&#xff0c;如何让人物随着鼠标滑动而转向&#xff1f; 方法 using System.Collections; using System.Collections.Generic; using UnityEngine; using UnityEngine.EventSystems;public class PlayerMovement : MonoBehaviour {public float speed 6f; …

STM32之IIC

(24条消息) C51之olde显示屏模块的使用_c51 oled_陈学弟的博客-CSDN博客IIC介绍 (24条消息) C51之olde显示屏模块的使用_c51 oled_陈学弟的博客-CSDN博客 函数封装 用到的库函数&#xff1a; HAL_StatusTypeDef HAL_I2C_Mem_Write ( I2C_HandleTypeDef * hi2c , uint16_t…

Day 1 认识软件测试——(软件测试定义、目的、原则)

Day 1 认识软件测试——(软件测试定义、目的、原则) 文章目录 Day 1 认识软件测试——(软件测试定义、目的、原则)软件测试的定义软件测试的目的软件测试的经济学问题黑盒测试白盒测试软件测试原则小结所谓软件测试,就是一个过程或一系列过程,用来确定计算机代码完成了其…

LitCTF 2023 WriteUp(部分)

Index 前言题目Web我Flag呢&#xff1f; Pwn只需要nc一下~口算题卡题目分析EXP: 狠狠的溢出涅~题目分析EXP: ezlogin题目分析EXP: Reverse世界上最棒的程序员 ez_XOREXP: CryptoHex&#xff1f;Hex&#xff01;(初级)梦想是红色的原来你也玩原神 Misc签到&#xff01;(初级)Wha…

【MyBatis】Spring Boot整合MyBatis操作数据库

文章目录 1. MyBatis是什么&#xff1f;2. 为什么要学MaBatis&#xff1f;3. MyBatis环境搭建4. MyBatis的使用4.1 简单示例4.2 获取动态参数4.2.1 ${xxx}获取动态参数4.2.2 #{xxx}获取动态参数4.2.3 #{xxx}与${xxx}获取字符串类型数据4.2.4 sql注入问题4.2.5 模糊查询like4.2.…

“Shell“脚本命令

文章目录 一.sort命令二.uniq命令三.tr命令四.cut命令五.split命令六.paste命令七.eval命令 一.sort命令 以行为单位对文件内容进行排序&#xff0c;也可以根据不同的数据类型来排序&#xff1b;比较原则是从首字符向后&#xff0c;依次按ASCII码值进行比较&#xff0c;最后将…

“Shell“数组

文章目录 一.数组1.1数组定义的方法1.2示例在这里插入图片描述 二.数组包2.1数组包括的数据类型2.2传输数组2.3返回数组 一.数组 1.1数组定义的方法 &#xff08;30 20 10 60 50 40&#xff09; //以空格相间隔的数组0 1 2 3 4 5 //下标方法一&am…