Go语言基础(四)—— 优质的容错处理

前言:
本专题用于记录自己(647)在Go语言方向的学习和积累。
系列内容比较偏基础,推荐给想要入门Go语言开发者们阅读。

目录如下:
Go语言基础(一)—— 简介、环境配置、HelloWorld
Go语言基础(二)—— 基本常用语法
Go语言基础(三)—— 面向对象编程
Go语言基础(四)—— 优质的容错处理
Go语言基础(五)—— 并发编程
Go语言基础(六)—— 测试、反射、Unsafe
Go语言基础(七)—— 架构 & 常见任务
Go语言基础(八)—— 性能调优


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

接下来,我们来看看在Go语言中如何做容错处理。

一、Go中的容错处理

  • 首先,我们要知道:Go语言中没有try/catch异常机制。

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

type error interface {
	Error() string
}
  • 通过errors.New快速创建error实例。
var xxxError = errors.New("xxxxx") // 快速创建错误类型

接下来举一个例子:
我们把之前写的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)
	}
}

二、panic、recover、os.Exit

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

我们举个简单的例子:

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

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

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

问:panicos.Exit究竟有什么区别呢?
1.os.Exit退出程序时不会先调用defer func代码块。
2.os.Exit退出程序时不会输出当前调用栈信息。

那么,如果我们就是想让程序不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!"))
}

因此,小心使用recover
可能会导致:

  1. 形成僵尸服务进程,使安全检查health check失效。
  2. 因为没有crash,导致提供不确定的服务。

因此,需要谨慎使用recover


最后,本系列我是在蔡超老师的技术分享下总结、实战完成的,
感谢蔡超老师的技术分享

PS:另附上,分享链接:《Go语言从入门到实战》
祝大家学有所成,工作顺利。谢谢!

https://juejin.im/post/5e37dffe5188252c5232b201

「点点赞赏,手留余香」

    还没有人赞赏,快来当第一个赞赏的人吧!
0 条回复 A 作者 M 管理员
    所有的伟大,都源于一个勇敢的开始!
欢迎您,新朋友,感谢参与互动!欢迎您 {{author}},您在本站有{{commentsCount}}条评论