golang语言错误处理方式check函数,把错误转化为panic

golang
0 69
suoniao 2021-01-20
需要:0索币

在看《Go入门指南》的一种用闭包处理错误的模式时,里面提到了一种错误的优雅处理方式,减少我们重复写if err:=f(); err != nil{}式的代码,感觉很心动,做了下测试,结论如下:

  1. 能减少if err式的代码,代码可以变清新整洁。
  2. 使用存在限制:只有当错误需要结束调用时才可以使用这种方法,如果被调用函数返回错误,但调用者函数需处理错误后,向下继续执行,则不能采用这种方法。

经常的写法

我们在设计函数时,错误处理要遵循以下2个规则:

  1. 被调用函数如果有错误,需要传递给调用者,一定要返回
  2. 调用者收到返回的错误,一定不可忽视,忽视就是埋bug。如果调用者不处理,被调函数就需要设计成无错误返回。

我们都有这种感受,一个函数需要调用许多函数,然后处理他们的错误,光if err就写了一堆,比如下面的test()函数。

package main

import (
	"errors"
	"log"
)

func main() {
	test()
}

func test() {
	if err := a(); err != nil {
		log.Println(err)
	}

	if err := b(); err != nil {
		log.Println(err)
	}

	if _, err := c(1, 0); err != nil {
		log.Println(err)
	}

	if _, err := d(1, 0); err != nil {
		log.Println(err)
	}
}

func a() error {
	return errors.New("error in a")
}

func b() error {
	return errors.New("error in b")
}

func c(x, y int) (int, error) {
	return x + y, errors.New("error in c")
}

func d(x, y int) (int, error) {
	if y == 0 {
		return 0, errors.New("error in d, divided by 0")
	}
	return x / y, nil
}

测试输出:

2018/10/24 14:42:40 error in a
2018/10/24 14:42:40 error in b
2018/10/24 14:42:40 error in c
2018/10/24 14:42:40 error in d, divided by 0

优雅的方式

借用2个小工具:

  1. check函数,把错误转化为panic
  2. 函数在defer中增加错误处理,从panic中恢复错误,并打印

我们对test()函数进行小小的改造。

package main

import (
	"errors"
	"fmt"
)

func main() {
	test()
}

func test() {
	defer func() {
		if r := recover(); r != nil {
			log.Println("got error: ", r)
		}
	}()

	var err error
	err = a()
	check(err)

	err = b()
	check(err)

	_, err = c(1, 2)
	check(err)

	_, err = d(1, 0)
	check(err)
}

func check(err error) {
	if err != nil {
		panic(err)
	}
}

func a() error {
	return errors.New("error in a")
}

func b() error {
	return errors.New("error in b")
}

func c(x, y int) (int, error) {
	return x + y, errors.New("error in c")
}

func d(x, y int) (int, error) {
	if y == 0 {
		return 0, errors.New("error in d, divided by 0")
	}
	return x / y, nil
}


输出结果:
```go
2018/10/24 17:29:35 got error:  error in a

test()函数是清爽了不少,也不用一直if然后处理err了,但是这种处理把错误转化为panic,导致test()后续的代码无法再继续执行,b(), c(), d()几个函数就没法执行了。

所以如果test()函数,遇到错误后就返回,就很适合这种优雅的方式。

还可以更优雅吗

test()函数中的defer看着挺碍眼的,我们还能让test()更简洁点吗,代码再优雅点?

采用文章中介绍的办法,增加errorHandler()函数,实现对被调用函数的封装,为它增加defer函数,恢复panic报的错误,看代码。

package main

import (
	"errors"
	"log"
)

func main() {
	errorHandler(test)()
}

func test() {
	var err error
	err = a()
	check(err)

	err = b()
	check(err)

	_, err = c(1, 2)
	check(err)

	_, err = d(1, 0)
	check(err)
}

func check(err error) {
	if err != nil {
		panic(err)
	}
}

// 封装f:为传入的函数增加defer
func errorHandler(f func()) func() {
	return func() {
		defer func() {
			if r := recover(); r != nil {
				log.Println("got error: ", r)
			}
		}()

		f()
	}
}

func a() error {
	return errors.New("error in a")
}

func b() error {
	return errors.New("error in b")
}

func c(x, y int) (int, error) {
	return x + y, errors.New("error in c")
}

func d(x, y int) (int, error) {
	if y == 0 {
		return 0, errors.New("error in d, divided by 0")
	}
	return x / y, nil
}

结果:

2018/10/24 17:36:41 got error:  error in a

还能再一次优雅吗

errorHandler()函数不够通用,它只接受无入参,无返回的函数。

  1. 如果这篇文章对你有帮助,不妨关注下我的Github,有文章会收到通知。
  2. 本文作者:大彬
  3. 如果喜欢本文,随意转载,但请保留此原文链接:http://lessisbetter.site/2018/10/24/go-handle-error/
回帖
  • 消灭零回复
广告位招租 100元/月 广告位招租 100元/月
相关主题
golang实现内存池 go语言字节池byte pool实现代码 0
golang网络编程之基于TCP协议实现长连接 golang心跳检测 0
golang类型断言type的使用 0
golang依赖包管理 mod使用教程 0
golang表单验证库validator 0
如何使用Go语言实现一个简单的异步任务框架呢?生产者消费者模型 0
golang 网络编程设置keepAlive空闲多长时间开始探测、 探测总次数 0
golang网络编程之TCP编程详解 0
go语言利用ioutil.ReadAll读取TCP socket所有数据 0
golang利用io.copy和bytes.Buffer读取TCP socket所有数据 0
golang读取所有socket数据的方式一 0
围绕Handler接口的方法ServeHTTP,可以轻松的写出go中的中间件 0
golang语言错误处理方式check函数,把错误转化为panic 0
golang cannot find module providing package 0
golang语言错误处理errors包使用详解 0
golang基于通道channel实现一个通用连接池 pool 0
golang数据类型之map结构详解 0
golang socket关闭读导致 wsarecv: An existing connection was forcibly closed by the remote host 0
windows配置golang的环境变量 gopath配置学习go语言第一天 0
golang应该是可以代替对使用c++存在误解的人吧,代替语言是不可能的 0
相关主题
索鸟快传2.1.0版本发布 0
golang网络数据读取timeout的处理 – 以SetReadDeadline为例 0
索鸟网广告位招租 0
golang利用通道chan实现一个通用的TCP连接池 0
frpc和frps的交互流程详解了解了frp是如何进行TCP代理的 0
golang实现内存池 go语言字节池byte pool实现代码 0
golang网络编程之基于TCP协议实现长连接 golang心跳检测 0
国产光刻机产业链最新核心材料ArF 193nm光刻胶通过用户认证 0
javascript数组相关函数length属性、delete关键字、pop()栈方法、shift() 0
Qt为我们提供了几个可以用于线程Sleep的函数msleep和usleep 0
QT程序闪退错误捕获教程利用DbgHelp 错误调试技术 0
简单快速修改Qtcreator项目工程的名称 0
golang 的interface接口类型断言 0
golang类型断言type的使用 0
golang依赖包管理 mod使用教程 0
golang表单验证库validator 0
如何使用Go语言实现一个简单的异步任务框架呢?生产者消费者模型 0
内网穿透反向代理工具frp实现TCP协议代理源码分析 0
内网穿透反向代理库frp的实现原理分析一 0
什么是socks5?socks5 是一个简单的代理协议 0