当心!recover成为”恶魔“--Go中的容错处理进阶

2024-01-16 18:12

本文主要是介绍当心!recover成为”恶魔“--Go中的容错处理进阶,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言:

Go语言本身没有try/catch异常机制,因为Go的三位创始人在设计Go语言之出觉得这样写会变得很繁琐。
但因为:Go本身支持函数多返回值,因此在写函数的时候,可以优先考虑容错处理。
接下来,我们来看看在Go语言中如何做容错处理。

一、Go中的容错处理

  • 1、Go语言中没有try/catch异常机制。

  • 2、要实现容错处理:使用error类型即可,默认实现error接口。

type error interface {Error() string
}
  • 3、通过errors.New快速创建error实例。

var xxxError = errors.New("xxxxx") // 快速创建错误类型

4、举一个例子:

我们把之前写的Fibonacci的例子加上容错处理,就变成了下面这样。

函数添加了多返回值,最后一个返回error。
若error有值,说明有异常;
若error无值,说明程序正常。

var LessThanTwoError = errors.New("n shoule not less than 2") // 定义错误类型func GetFibonacci(n int) ([]int, error) {// 容错处理if n <= 2 {return nil, LessThanTwoError}fibList := []int{1, 1}for i := 2; i < n; i++ {fibList = append(fibList, fibList[i-2]+fibList[i-1])}return fibList, nil
}func TestGetFibonacci(t *testing.T) {if value, err := GetFibonacci(0); err != nil {if err == LessThanTwoError {fmt.Println("It is less error.")}t.Error(err)} else {t.Log(value)}
}

代码运行结果如下:

=== RUN   TestGetFibonacci
It is less error.
    recover_test.go:65: n shoule not less than 2
--- FAIL: TestGetFibonacci (0.00s)

FAIL

二、panic、recover、os.Exit

  • panic:用于发送不可恢复的错误,执行defer func内的代码块,并请求退出程序。

  • recover:用于恢复panic抛出的错误。

  • os.Exit:用于直接退出程序。

1、举个panic的例子

func TestPanic(t *testing.T) {defer func() {fmt.Println("Finally!")}()fmt.Println("Test panic is Started.")panic(errors.New("Something wrong!"))
}

测试不通过且抛出异常,输出如下:

=== RUN   TestPanic
Test panic is Started.
Finally!
--- FAIL: TestPanic (0.00s)
panic: Something wrong! [recovered]
    panic: Something wrong!

goroutine 19 [running]:
testing.tRunner.func1.2({0xe195a0, 0xc00008a250})
    D:/Program Files/Go/src/testing/testing.go:1545 +0x238
testing.tRunner.func1()
    D:/Program Files/Go/src/testing/testing.go:1548 +0x397
panic({0xe195a0?, 0xc00008a250?})
    D:/Program Files/Go/src/runtime/panic.go:914 +0x21f
command-line-arguments.TestPanic(0x0?)
    D:/golang/ch14/recover_test.go:26 +0x9d
testing.tRunner(0xc000084820, 0xe47e38)
    D:/Program Files/Go/src/testing/testing.go:1595 +0xff
created by testing.(*T).Run in goroutine 1
    D:/Program Files/Go/src/testing/testing.go:1648 +0x3ad


进程 已完成,退出代码为 1

2、举个os.Exit的例子

其实,os.Exit也可以退出程序。

func TestOsExit(t *testing.T) {fmt.Println("Test os.Exit is Started.")os.Exit(0)
}

输出如下:

=== RUN   TestOsExit
Test os.Exit is Started.
--- FAIL: TestOsExit (0.00s)
panic: unexpected call to os.Exit(0) during test [recovered]
    panic: unexpected call to os.Exit(0) during test

goroutine 6 [running]:
testing.tRunner.func1.2({0x7036a0, 0x75c0a0})
    D:/Program Files/Go/src/testing/testing.go:1545 +0x238
testing.tRunner.func1()
    D:/Program Files/Go/src/testing/testing.go:1548 +0x397
panic({0x7036a0?, 0x75c0a0?})
    D:/Program Files/Go/src/runtime/panic.go:914 +0x21f
os.Exit(0x0)
    D:/Program Files/Go/src/os/proc.go:67 +0x51
command-line-arguments.TestOsExit(0x0?)
    D:/golang/ch14/recover_test.go:42 +0x53
testing.tRunner(0xc000045520, 0x737e78)
    D:/Program Files/Go/src/testing/testing.go:1595 +0xff
created by testing.(*T).Run in goroutine 1
    D:/Program Files/Go/src/testing/testing.go:1648 +0x3ad


进程 已完成,退出代码为 1

3、思考:panicos.Exit究竟有什么区别?

os.Exit退出程序时不会先调用defer func代码块。
os.Exit退出程序时不会输出当前调用栈信息。

4、使用recover举例

那么,如果我们就是想让程序不crash,有没有办法呢?

答案是有的,使用recover,但是很不推荐这么使用recover
因为并没有解决发生panic的问题,只是把错误移除,这样是很不安全的。
甚至,如果是因为系统资源panic,这样我们的服务就变成了僵尸服务,虽然活着但无法提供服务功能。

recover使用方式如下,但一般不推荐使用。

func TestPanicRecover(t *testing.T) {defer func() {if err := recover(); err != nil { // 恢复错误fmt.Println("recover panic", err)}}()fmt.Println("Test panic is Started.")panic(errors.New("Something wrong!"))
}

测试居然通过了,输出如下:

=== RUN   TestPanicRecover
Test panic is Started.
recover panic Something wrong!
--- PASS: TestPanicRecover (0.00s)
PASS

5、必须要小心使用recover

因为使用recover可能会导致:

  1. 形成僵尸服务进程,使安全检查health check失效。

  2. 因为没有crash,导致提供不确定的服务。

所以“Let it Crash!”,让程序异常时通过重启来恢复而不是通过recover跳过异常,往往是我们恢复不确定性错误的最好方法!

这篇关于当心!recover成为”恶魔“--Go中的容错处理进阶的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/613464

相关文章

Python进阶之Excel基本操作介绍

《Python进阶之Excel基本操作介绍》在现实中,很多工作都需要与数据打交道,Excel作为常用的数据处理工具,一直备受人们的青睐,本文主要为大家介绍了一些Python中Excel的基本操作,希望... 目录概述写入使用 xlwt使用 XlsxWriter读取修改概述在现实中,很多工作都需要与数据打交

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

Go语言使用Buffer实现高性能处理字节和字符

《Go语言使用Buffer实现高性能处理字节和字符》在Go中,bytes.Buffer是一个非常高效的类型,用于处理字节数据的读写操作,本文将详细介绍一下如何使用Buffer实现高性能处理字节和... 目录1. bytes.Buffer 的基本用法1.1. 创建和初始化 Buffer1.2. 使用 Writ

Python视频处理库VidGear使用小结

《Python视频处理库VidGear使用小结》VidGear是一个高性能的Python视频处理库,本文主要介绍了Python视频处理库VidGear使用小结,文中通过示例代码介绍的非常详细,对大家的... 目录一、VidGear的安装二、VidGear的主要功能三、VidGear的使用示例四、VidGea

Python结合requests和Cheerio处理网页内容的操作步骤

《Python结合requests和Cheerio处理网页内容的操作步骤》Python因其简洁明了的语法和强大的库支持,成为了编写爬虫程序的首选语言之一,requests库是Python中用于发送HT... 目录一、前言二、环境搭建三、requests库的基本使用四、Cheerio库的基本使用五、结合req

使用Python处理CSV和Excel文件的操作方法

《使用Python处理CSV和Excel文件的操作方法》在数据分析、自动化和日常开发中,CSV和Excel文件是非常常见的数据存储格式,ython提供了强大的工具来读取、编辑和保存这两种文件,满足从基... 目录1. CSV 文件概述和处理方法1.1 CSV 文件格式的基本介绍1.2 使用 python 内

Go Gorm 示例详解

《GoGorm示例详解》Gorm是一款高性能的GolangORM库,便于开发人员提高效率,本文介绍了Gorm的基本概念、数据库连接、基本操作(创建表、新增记录、查询记录、修改记录、删除记录)等,本... 目录1. 概念2. 数据库连接2.1 安装依赖2.2 连接数据库3. 数据库基本操作3.1 创建表(表关

如何使用celery进行异步处理和定时任务(django)

《如何使用celery进行异步处理和定时任务(django)》文章介绍了Celery的基本概念、安装方法、如何使用Celery进行异步任务处理以及如何设置定时任务,通过Celery,可以在Web应用中... 目录一、celery的作用二、安装celery三、使用celery 异步执行任务四、使用celery

SpringBoot操作spark处理hdfs文件的操作方法

《SpringBoot操作spark处理hdfs文件的操作方法》本文介绍了如何使用SpringBoot操作Spark处理HDFS文件,包括导入依赖、配置Spark信息、编写Controller和Ser... 目录SpringBoot操作spark处理hdfs文件1、导入依赖2、配置spark信息3、cont

Go信号处理如何优雅地关闭你的应用

《Go信号处理如何优雅地关闭你的应用》Go中的优雅关闭机制使得在应用程序接收到终止信号时,能够进行平滑的资源清理,通过使用context来管理goroutine的生命周期,结合signal... 目录1. 什么是信号处理?2. 如何优雅地关闭 Go 应用?3. 代码实现3.1 基本的信号捕获和优雅关闭3.2