golang error

type error interface {
    Error() string
}

内建error接口类型是约定用于表示错误信息,nil值表示无错误。

package errors
import “errors”
errors包实现了创建错误值的函数。

package main

import (
	"fmt"
	"time"
)

// MyError is an error implementation that includes a time and message.
type MyError struct {
	When time.Time
	What string
}

func (e MyError) Error() string {	// 实现了Error接口
	return fmt.Sprintf("%v: %v", e.When, e.What)
}
func oops() error {
	return MyError{
		time.Date(1989, 3, 15, 22, 30, 0, 0, time.UTC),
		"the file system has gone away",
	}
}

func main() {
	if err := oops(); err != nil {
		fmt.Println(err)
	}
}

运行结果:1989-03-15 22:30:00 +0000 UTC: the file system has gone away

func New(text string) error

使用字符串创建一个错误,请类比fmt包的Errorf方法,差不多可以认为是New(fmt.Sprintf(…))。

package main

import (
	"errors"
	"fmt"
)

func main() {
	err := errors.New("emit macho dwarf: elf header corrupted")
	if err != nil {
		fmt.Print(err)
	}
}

运行结果:emit macho dwarf: elf header corrupted

package main

import (
	"fmt"
)

func main() {
	const name, id = "bimmler", 17
	err := fmt.Errorf("user %q (id %d) not found", name, id)
	if err != nil {
		fmt.Print(err)
	}
}

运行结果:user “bimmler” (id 17) not found

fmt.Errorf 和 errors.New的返回值

package main
import (
	"errors"
	"fmt"
)
func test1() error {
	return fmt.Errorf("aaaaa")
}
func test2() error {
	return errors.New("bbbbb")
}
func main() {
	err := test1()
	fmt.Printf("%T : %v\n", err, err)
	err = test2()
	fmt.Printf("%T : %v\n", err, err)
}

运行结果:
*errors.errorString : aaaaa
*errors.errorString : bbbbb

对error信息进行封装

package main

import (
	"errors"
	"fmt"
	"strings"
)

type decodeError struct {
	what  error
	stack []string
}

func wrapError(err error, ctx string) error {
	if err == nil {
		return nil
	}

	// 如果传入的类型为 decodeError 将其内容追加到 stack 里
	if decErr, ok := err.(*decodeError); ok {
		decErr.stack = append(decErr.stack, ctx)
		return decErr
	}

	// 如果传入的err类型不是decodeError类型,将其保存为decodeError类型
	return &decodeError{err, []string{ctx}}
}

// 实现error的 Error 接口,该接口会在fmt.Println的时候调用。
// err.what 保存了 errors.New 的错误信息。
// strings.Join 是将 err.stack 中的字符串用 <- 连接起来。
func (err *decodeError) Error() string {
	return fmt.Sprintf("%v (decode path: %s)", err.what, strings.Join(err.stack, "<-"))
}

func test() error {
	return errors.New("aaaaa")
}

func main() {

	err := test()
	err = wrapError(err, "mmmmm")
	err = wrapError(err, "nnnnn")
	err = wrapError(err, "lllll")
	fmt.Println(err)
}

运行结果:aaaaa (decode path: mmmmm<-nnnnn<-lllll)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Go 语言中,错误处理是一种非常重要的机制,可以帮助我们在程序运行时检测到错误,并采取相应的措施来解决问题。Go 语言中的错误类型是内置的 error 接口类型,其定义如下: ```go type error interface { Error() string } ``` 可以看到,这个接口只有一个方法 Error(),它返回一个字符串,表示错误的描述信息。因此,如果你想定义一个错误类型,只需要实现 error 接口的 Error() 方法即可。 在 Go 语言中,我们通常会使用函数返回值来传递错误信息。如果函数执行成功,通常会返回一个 nil 错误;如果函数执行失败,通常会返回一个非空的错误值,表示出错的具体信息。 例如,下面的代码演示了如何在打开一个文件时进行错误处理: ```go file, err := os.Open("test.txt") if err != nil { // 处理错误 fmt.Println("打开文件失败:", err.Error()) return } // 文件打开成功,进行操作 ``` 在这个代码中,我们使用 os.Open 函数打开一个文件。如果函数执行失败,会返回一个非空的错误值 err;否则,函数返回一个文件对象 file。我们可以使用 if err != nil 来检查 err 是否为空,如果不为空,表示函数执行失败,我们需要采取相应的措施来解决问题,例如打印错误信息并返回。如果 err 为空,则表示函数执行成功,我们可以继续操作文件对象。 除了使用 if err != nil 来检查错误之外,Go 语言还提供了一个更加简洁的语法,即使用 defer 和 panic 函数来处理错误。例如,下面的代码演示了如何在除数为零时触发 panic: ```go func divide(x, y int) int { defer func() { if err := recover(); err != nil { fmt.Println("出现了一个错误:", err) } }() if y == 0 { panic("除数不能为零") } return x / y } ``` 在这个代码中,我们使用 defer 和匿名函数来定义一个错误处理函数。在 divide 函数中,如果除数为零,我们会使用 panic 函数触发一个 panic,表示程序遇到了无法处理的错误。此时,defer 语句会立即执行匿名函数,该函数调用 recover 函数来捕获 panic,并打印错误信息。注意,在 defer 函数中使用 recover 函数可以避免程序崩溃,并返回一个错误信息。 总的来说,Go 语言提供了多种方式来处理错误,可以根据实际情况选择适合自己的方式。在实际开发中,我们通常会将错误信息记录到日志中,或者通过 HTTP 接口返回给客户端。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值