GO 错误处理


defer+recover机制处理错误

(1)我们来写一个错误

package main

import "fmt"

func main() {
	test()
	fmt.Println("上面的除法操作执行成功...")
	fmt.Println("正常执行下面的逻辑...")
}
func test() {
	num1 := 10
	num2 := 0
	result := num1 / num2   //第13行
	fmt.Println(result)
}

我们来运行一下,结果如图:
在这里插入图片描述
意思是(死机:运行时错误:整数被零除)

发现:程序中出现错误/恐慌以后,程序被中断,无法继续执行。

(2)错误处理/捕获机制
GO中追求代码优雅,引入机制:defer+recover机制处理错误 。
内置函数recover:

func recover() interface{}

内建函数recover允许程序管理恐慌过程中的GO程。在defer的函数中,执行recover调用会取回传至panic调用的错误值,恢复正常执行,停止恐慌过程。若recover在defer的函数之外被调用,它将不会停止恐慌过程序列。在此情况下,或当该Go程不在恐慌过程中时,或提供给panic的实参为nil时,recover就会返回nil。

package main

import "fmt"

func main() {
	test()
	fmt.Println("上面的除法操作执行成功...")
	fmt.Println("正常执行下面的逻辑...")
}
func test() {
	//利用defer+recover来捕获错误:defer后加上匿名函数的调用
	defer func() {
		//调用recover内置函数,可以捕获错误:
		err := recover()
		//如果没有捕获错误,返回值为零值:nil
		if err != nil {
			fmt.Println("捕获到错误")
			fmt.Println("err是:", err)
		}
	}()

	num1 := 10
	num2 := 0
	result := num1 / num2
	fmt.Println(result)
}

在这里插入图片描述

优点:提高程序健壮性

自定义错误

自定义错误:需要调用errors包下的New函数:函数返回error类型

func New(text string) error

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

package main

import (
	"errors"
	"fmt"
)

func main() {
	err := test()
	if err != nil {
		fmt.Println("自定义错误:", err)
	}
	fmt.Println("上面的除法操作执行成功...")
	fmt.Println("正常执行下面的逻辑...")
}
func test() (err error) {
	num1 := 10
	num2 := 0
	if num2 == 0 {
		//抛出自定义错误:
		return errors.New("除数不能为0喔")
	} else { //如果除数不为0,那么正常执行就可以了
		result := num1 / num2
		fmt.Println(result)
		//如果没有错误,返回零值:
		return nil
	}
}

运行结果:

在这里插入图片描述
有一种情况:程序出现错误以后,后续代码就没有必要执行,想让程序中断,退出程序:

借助:builtin包下内置函数:panic

func panic(v interface{})

内建函数panic停止当前Go程的正常执行。当函数F调用panic时,F的正常执行就会立刻停止。F中defer的所有函数先入后出执行后,F返回给其调用者G。G如同F一样行动,层层返回,直到该Go程中所有函数都按相反的顺序停止执行。之后,程序被终止,而错误情况会被报告,包括引发该恐慌的实参值,此终止序列称为恐慌过程。

package main
import (
	"errors"
	"fmt"
)

func main() {
	err := test()
	if err != nil {
		fmt.Println("自定义错误:", err)
panic(err)
	}
	fmt.Println("上面的除法操作执行成功...")
	fmt.Println("正常执行下面的逻辑...")
}
func test() (err error) {
	num1 := 10
	num2 := 0
	if num2 == 0 {
		//抛出自定义错误:
		return errors.New("除数不能为0喔")
	} else { //如果除数不为0,那么正常执行就可以了
		result := num1 / num2
		fmt.Println(result)
		//如果没有错误,返回零值:
		return nil
	}
}

运行结果:
在这里插入图片描述

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值