当心!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 Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

一文详解Java异常处理你都了解哪些知识

《一文详解Java异常处理你都了解哪些知识》:本文主要介绍Java异常处理的相关资料,包括异常的分类、捕获和处理异常的语法、常见的异常类型以及自定义异常的实现,文中通过代码介绍的非常详细,需要的朋... 目录前言一、什么是异常二、异常的分类2.1 受检异常2.2 非受检异常三、异常处理的语法3.1 try-

Python使用getopt处理命令行参数示例解析(最佳实践)

《Python使用getopt处理命令行参数示例解析(最佳实践)》getopt模块是Python标准库中一个简单但强大的命令行参数处理工具,它特别适合那些需要快速实现基本命令行参数解析的场景,或者需要... 目录为什么需要处理命令行参数?getopt模块基础实际应用示例与其他参数处理方式的比较常见问http

Java Response返回值的最佳处理方案

《JavaResponse返回值的最佳处理方案》在开发Web应用程序时,我们经常需要通过HTTP请求从服务器获取响应数据,这些数据可以是JSON、XML、甚至是文件,本篇文章将详细解析Java中处理... 目录摘要概述核心问题:关键技术点:源码解析示例 1:使用HttpURLConnection获取Resp

Java中Switch Case多个条件处理方法举例

《Java中SwitchCase多个条件处理方法举例》Java中switch语句用于根据变量值执行不同代码块,适用于多个条件的处理,:本文主要介绍Java中SwitchCase多个条件处理的相... 目录前言基本语法处理多个条件示例1:合并相同代码的多个case示例2:通过字符串合并多个case进阶用法使用

Java实现优雅日期处理的方案详解

《Java实现优雅日期处理的方案详解》在我们的日常工作中,需要经常处理各种格式,各种类似的的日期或者时间,下面我们就来看看如何使用java处理这样的日期问题吧,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言一、日期的坑1.1 日期格式化陷阱1.2 时区转换二、优雅方案的进阶之路2.1 线程安全重构2

Python处理函数调用超时的四种方法

《Python处理函数调用超时的四种方法》在实际开发过程中,我们可能会遇到一些场景,需要对函数的执行时间进行限制,例如,当一个函数执行时间过长时,可能会导致程序卡顿、资源占用过高,因此,在某些情况下,... 目录前言func-timeout1. 安装 func-timeout2. 基本用法自定义进程subp

Java字符串处理全解析(String、StringBuilder与StringBuffer)

《Java字符串处理全解析(String、StringBuilder与StringBuffer)》:本文主要介绍Java字符串处理全解析(String、StringBuilder与StringBu... 目录Java字符串处理全解析:String、StringBuilder与StringBuffer一、St

go 指针接收者和值接收者的区别小结

《go指针接收者和值接收者的区别小结》在Go语言中,值接收者和指针接收者是方法定义中的两种接收者类型,本文主要介绍了go指针接收者和值接收者的区别小结,文中通过示例代码介绍的非常详细,需要的朋友们下... 目录go 指针接收者和值接收者的区别易错点辨析go 指针接收者和值接收者的区别指针接收者和值接收者的

浅析Java中如何优雅地处理null值

《浅析Java中如何优雅地处理null值》这篇文章主要为大家详细介绍了如何结合Lambda表达式和Optional,让Java更优雅地处理null值,感兴趣的小伙伴可以跟随小编一起学习一下... 目录场景 1:不为 null 则执行场景 2:不为 null 则返回,为 null 则返回特定值或抛出异常场景