重学Go语言 | Go创建error的几种方式

公众号:程序员读书,欢迎关注

Go语言函数(或方法)是支持多个返回值的,因此在Go语言的编程哲学中,函数的返回值的最后一个通常都是error类型:

go 复制代码
//标准os包下的函数
func Open(name string) (*File, error) {
	return OpenFile(name, O_RDONLY, 0)
}

error类型作为函数的最后一个返回值,其作用在于告诉调用者函数在执行过程是否发生错误,当返回的error值为nil,表示函数正常执行。

其实,error类型本质上是一个Go内置的接口,其定义如下:

go 复制代码
type error interface {
	Error() string
}

从上面error的定义可以看到error接口只有一个Error()方法。

遵循着Go语言的编程哲学,当我们开发自己的函数时最好也把error作为函数的最后一个返回值。

即然要返回error类型,那么就必须先创建error类型,怎么创建呢?在这篇文章我们简单学习三种创建error类型的方式。

使用errors包

Go标准库errors包为了我们提供了一个最简单的error接口实现:

go 复制代码
//errors包errors.go文件
type errorString struct {
	s string
}

func (e *errorString) Error() string {
	return e.s
}

从上面的定义可以看到errorString实现了error接口,且参数非常简单,只有一个文本信息,通过errors.New()函数可以创建一个errorString类型:

go 复制代码
package school

import "errors"

type Student struct {
	ID    int
	Name  string
	Score int
}

func NewStudent(id int, name string, score int) (*Student, error) {
	if id <= 0 {
		return nil, errors.New("id could not less than 0")
	}

	if score < 0 || score > 100 {
		return nil, errors.New("score must between 0 and 100")
	}

	return &Student{id, name, score}, nil
}

fmt.Errorf()函数

有时候我们需要复杂的错误信息,如果直接调用errors.New()方法的话,需要自己拼接字符串,而标准库fmt包的Errorf()函数可以格式化文本并返回一个error类型:

go 复制代码
package school

import (
	"fmt"
)

type Student struct {
	ID    int
	Name  string
	Score int
}

func NewStudent(id int, name string, score int) (*Student, error) {
	if id <= 0 {
		return nil, fmt.Errorf("%d is less than 0", id)
	}

	if score < 0 || score > 100 {
		return nil, fmt.Errorf("%d is not between 0 and 100", score)
	}

	return &Student{id, name, score}, nil
}

自己实现error接口

上述的两种创建error类型方式,本质上包内部有一个error接口的实现,所以如果我们希望返回更详细的错误类型,方便调用者判断,可以实现自己的错误类型:

go 复制代码
package main

import (
	"fmt"
	"net/http"
)

type ResponseError struct {
	code    int
	message string
}

func (r ResponseError) Error() string {
	return fmt.Sprintf(`{"code":%d,"message":"%s"}`, r.code, r.message)
}

func getUser(id string) (map[string]string, error) {
	if len(id) == 0 {
		return nil, &ResponseError{1, "id不能为空"}
	}
	return map[string]string{"id": id, "name": "测试"}, nil
}

func main() {
	http.HandleFunc("/user", func(w http.ResponseWriter, r *http.Request) {
		id := r.FormValue("id")
		user, err := getUser(id)

		if err != nil {
			fmt.Fprint(w, err)
			return
		}
		fmt.Fprint(w, user)
	})
	http.ListenAndServe(":80", nil)
}

小结

error类型是Go语言的内置类型,其本质是一个只有Error()方法的接口,因此只要实现该方法就可以作为error类型使用。

相关推荐
KeepPush几秒前
Python JSON 完全指南:从基础到实战,掌握数据交换核心技能
后端
Cosolar13 分钟前
深度测评 | QoderWork:当 AI 不再只是"聊天搭子",而是真能帮你干活的桌面智能体
人工智能·后端·程序员
前端Hardy14 分钟前
21.8 万周下载!这个 React 表格组件,10 行代码就能跑起来
前端·javascript·后端
用户83562907805128 分钟前
使用 Python 在 Word 文档中添加和管理脚注
后端
字节跳动数据库1 小时前
一个请求稳定的一生
后端·程序员
RainCity1 小时前
Java Swing 自定义组件库分享(十一)
java·笔记·后端
用户743835613511 小时前
无锁 Hub:我的 IM 系统为什么用 channel 而不是 mutex 管理在线用户
go
掘金一周2 小时前
问卷调查:如果现在收到裁员通知,你手里的现金流能支撑多久? | 沸点周刊6.4
前端·人工智能·后端
JustHappy3 小时前
古法编程秘籍(四):函数究竟是什么?把函数最重要的能力一次讲清楚
前端·后端·面试
_Evan_Yao3 小时前
一文搞懂:Git分支管理与团队协作规范——从GitFlow到GitHub Flow,从rebase到merge,打造高效协作流
java·git·后端·github