go语言的异常处理机制

news2024/11/27 16:46:05

error

在go语言中,异常被定义为实现了error接口的类型,error接口只定义了一个返回string类型Error()方法,任何实现了Error()方法的类型都可以被定义为异常,以下是一个自定义的异常类型:
 

type MyError struct {
	data   string
	errNO  int
	errMsg string
}
// 定义错误方法
func (err *MyError) Error() string {
	return err.errMsg
}

如果我们直接打印一个error对象,实际上调用的是error.Error()

panic

在通常情况下,向程序使用方报告错误状态的方式可以是返回一个额外的error类型值。

但是,当遇到不可恢复的错误状态的时候,如数组访问越界、空指针引用等,这些运行时错误会引起painc异常。这时,上述错误处理方式显然就不适合了。反过来讲,在一般情况下,我们不应通过调用panic函数来报告普通的错误,而应该只把它作为报告致命错误的一种方式。当某些不应该发生的场景发生时,我们就应该调用panic。

一般而言,当panic异常发生时,程序会中断运行,并立即执行在该goroutine(可以先理解成线程,在中被延迟的函数(defer 机制)。随后,程序崩溃并输出日志信息。日志信息包括panic value和函数调用的堆栈跟踪信息。

不是所有的panic异常都来自运行时,直接调用内置的panic函数也会引发panic异常;panic函数接受任何值作为参数。

异常的捕获recover

在Go语言中,异常可以通过recover()函数来捕获。recover()函数必须在defer语句中调用,用于捕获当前函数调用栈中的异常。如果当前函数中没有异常,recover()函数将返回nil

以下是一个使用recover()函数捕获异常的示例:

func DoSomething() error {
    defer func() {
        if r := recover(); r != nil {
            fmt.Println(r)
        }
    }()
    if someError {
        panic(&MyException{"something went wrong"})
    }
    return nil
}
package main

import "fmt"

//type error interface {
//	Error() string
//}

// 自定义错误返回类型
type MyError struct {
	data   string
	errNO  int
	errMsg string
}

var defaultCode = 200

// 定义错误方法
func (err *MyError) Error() string {
	return err.errMsg
}

// 定义两种不同的方法返回函数
func defaultError(msg string) error {
	return &MyError{errMsg: msg, errNO: defaultCode}
}
func initError(msg string, errno int) error {
	return &MyError{errMsg: msg, errNO: errno}
}
func div(y int) int {
	var x = 10
	z := x / y
	fmt.Println(z)
	return z

}
func do() {

}
func main() {

	defer func() {
		if e := recover(); e != nil {
			err := initError("sda", 232)
			fmt.Println(err)
		}
	}()
	div(0)

}

在上面的示例中,我们在DoSomething()函数中使用了defer语句,在函数返回之前将一个匿名函数压入调用栈。这个匿名函数中使用了recover()函数来捕获异常。如果异常被捕获,就会输出异常信息,否则这个函数什么也不会做。

panicrecover的实现原理

在Go语言中,panicrecover语句的实现原理比较复杂。下面将介绍这些语句的实现原理。

  • panic的实现原理

当程序执行到panic函数时,它会停止当前的执行流程,并且开始执行所有被注册的defer语句。然后,程序会停止执行,并且将控制权交给调用栈中的上一级函数。如果调用栈中的任何函数都没有捕获这个异常,程序将会崩溃并且打印出错误信息。

在底层实现中,panic函数会创建一个panic结构体,并且将该结构体的指针保存在当前的goroutine结构体中。然后,panic函数会继续执行所有被注册的defer语句,直到所有的defer语句都执行完毕。最后,panic函数会停止执行,并且将控制权交给调用栈中的上一级函数。

  • recover的实现原理

当程序执行到recover函数时,它会检查当前的goroutine结构体中是否存在一个panic结构体的指针。如果存在,recover函数会返回该panic结构体的值,并且清除该结构体的指针。然后,程序会继续执行。

在底层实现中,recover函数会检查当前的goroutine结构体中是否存在一个panic结构体的指针。如果存在,recover函数会返回该panic结构体的值,并且清除该结构体的指针。如果不存在,recover函数会返回nil

异常处理的最佳实践

在Go语言中,异常处理是非常重要的,它可以帮助我们诊断程序中的错误,并提供一种优雅的方式来处理这些错误。以下是一些处理异常的最佳实践:

  1. 不要滥用异常

异常应该只用于处理真正的异常情况,例如不可恢复的错误、硬件故障等。不要滥用异常,将其作为一种流程控制的手段来使用。

2. 在需要的地方抛出异常

只有在必要的时候才应该抛出异常,例如当出现无法恢复的错误或者不符合预期的行为时。不要在可以通过其他方式解决的问题上抛出异常。

3. 在需要的地方捕获异常

只有在需要的时候才应该捕获异常。在不需要处理异常的情况下,不要捕获异常。在必要的时候,尽可能早地捕获异常,以避免异常在代码中蔓延。

4. 在捕获异常时提供上下文信息

当捕获异常时,应该提供足够的上下文信息,以便于快速定位和解决问题。例如,在异常信息中包含发生异常的函数、文件名、行号等信息,可以帮助我们快速定位问题所在。

5. 不要忽略异常

不要忽略任何异常,无论是自己抛出的还是调用库函数时捕获的异常。忽略异常会导致程序的行为不可预期,可能会导致严重的后果。

总结

Go语言异常机制是一种非常重要的特性,它可以帮助我们诊断程序中的错误,并提供一种优雅的方式来处理这些错误。在使用异常时,需要注意不要滥用异常,只在必要的时候抛出异常和捕获异常,提供足够的上下文信息,并且不要忽略任何异常。如果合理使用异常,可以让我们的代码更加健壮、清晰和可读。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1903288.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

28行代码完成深度学习模型——线性模型 01

在这里插入代码片## 线性模型 机器学习中的线性模型是一种预测模型,它基于线性关系来预测输出值。这种模型假设输入特征(自变量)和输出(因变量)之间存在线性关系。线性模型通常具有以下形式: y x*w b 其…

代码随想录算法训练营第二十七天|452. 用最少数量的箭引爆气球、435. 无重叠区间、763.划分字母区间

452. 用最少数量的箭引爆气球 如何使用最少的弓箭呢? 直觉上来看,貌似只射重叠最多的气球,用的弓箭一定最少,那么有没有当前重叠了三个气球,我射两个,留下一个和后面的一起射这样弓箭用的更少的情况呢&am…

生成式AI的短板在于“Token”的存在

生成式AI模型处理文本的方式与人类不同。理解它们基于“token”的内部环境,可能有助于解释一些奇怪行为和固有局限性。 从小型设备上的Gemma到OpenAI领先行业的GPT-4o,大多数模型都是基于一种称为Transformer的架构。由于Transformer在将文本与其他类型…

Golang | Leetcode Golang题解之第220题存在重复元素III

题目: 题解: func getID(x, w int) int {if x > 0 {return x / w}return (x1)/w - 1 }func containsNearbyAlmostDuplicate(nums []int, k, t int) bool {mp : map[int]int{}for i, x : range nums {id : getID(x, t1)if _, has : mp[id]; has {retu…

电商利器——淘宝商品月销量API接口解析

在电商时代,数据就是金钱。对于淘宝商家而言,掌握商品的销量数据无异于掌握了市场的脉搏。如今,淘宝商品月销量API接口的出现,联讯数据让商家如虎添翼,能够更加精准地把握市场动态,优化商品策略。 淘宝商…

Unity3D游戏 RPG

丛林探险游戏 人物进行探险游戏 拥有登录,首页,3D物体旋转浏览的功能,还能进行种植树等功能

[图解]SysML和EA建模住宅安全系统-12-内部块图

1 00:00:00,580 --> 00:00:02,770 接下来我们来画流了 2 00:00:03,100 --> 00:00:05,050 首先第一个是站点状态 3 00:00:05,140 --> 00:00:08,130 从这里到这里,我们画一个过来 4 00:00:10,290 --> 00:00:11,890 这里流到这里 5 00:00:11,900 -->…

[c++] 可变参数模版

前言 可变参数模板是C11及之后才开始使用,学校的老古董编译器不一定能用 相信大家在刚入门c/c时都接触过printf函数 int printf ( const char * format, ... ); printf用于将数据格式化输出到屏幕上,它的参数非常有意思,可以支持任意数量,任意类型的多参数.而如果我们想实现类…

算法系列--分治排序|归并排序|逆序对的求解

一.基本概念与实现 归并排序(mergeSort)也是基于分治思想的一种排序方式,思路如下: 分解:根据中间下标mid将数组分解为两部分解决:不断执行上述分解过程,当分解到只有一个元素时,停止分解,此时就是有序的合并:合并两个有序的子区间,所有子区间合并的结果就是原问题的解 归并…

基于java+springboot+vue实现的仓库管理系统(文末源码+lw+ppt)23-499

第1章 绪论 伴随着信息社会的飞速发展,仓库管理所面临的问题也一个接一个的出现,所以现在最该解决的问题就是信息的实时查询和访问需求的问题,以及如何利用快捷便利的方式让访问者在广大信息系统中进行查询、分享、储存和管理。这对我们的现…

QListWidget 缩略图IconMode示例

1、实现的效果如下&#xff1a; 2、实现代码 &#xff08;1&#xff09;头文件 #pragma once #include <QtWidgets/QMainWindow> #include "ui_QListViewDemo.h" enum ListDataType { ldtNone -1, ldtOne 0, ldtTwo 1, }; struct ListData…

系统化学习 H264视频编码(01)基础概念

说明&#xff1a;我们参考黄金圈学习法&#xff08;什么是黄金圈法则?->模型 黄金圈法则&#xff0c;本文使用&#xff1a;why-what&#xff09;来学习音H264视频编码。本系列文章侧重于理解视频编码的知识体系和实践方法&#xff0c;理论方面会更多地讲清楚 音视频中概念的…

读书笔记:终身成长

思维模式 两种思维模式对你意味着什么 相信自己的才能是一成不变的——也就是固定型的思维模式——会使你急于一遍遍地证明自己的能力。如果你只拥有一般水平的智力和品德&#xff0c;以及普通的个性——那么&#xff0c;你最好证明你自己能够在这些方面达到正常水平&#xf…

【Python】一文向您详细介绍 argparse中 action=‘store_true’ 的作用

【Python】一文向您详细介绍 argparse中 action‘store_true’ 的作用 下滑即可查看博客内容 &#x1f308; 欢迎莅临我的个人主页 &#x1f448;这里是我静心耕耘深度学习领域、真诚分享知识与智慧的小天地&#xff01;&#x1f387; &#x1f393; 博主简介&#xff1a;98…

自动控制:前馈控制

自动控制&#xff1a;前馈控制 前馈控制是一种在控制系统中通过预先计算和调整输入来应对已知扰动或变化的方法。相比于反馈控制&#xff0c;前馈控制能够更快速地响应系统的变化&#xff0c;因为它不依赖于系统输出的反馈信号。前馈控制的应用在工业过程中尤为广泛&#xff0…

DAMA学习笔记(四)-数据建模与设计

1.引言 数据建模是发现、分析和确定数据需求的过程&#xff0c;用一种称为数据模型的精确形式表示和传递这些数据需求。建模过程中要求组织发现并记录数据组合的方式。数据常见的模式: 关系模式、多维模式、面向对象模式、 事实模式、时间序列模式和NoSQL模式。按照描述详细程度…

染色の树-美团2023笔试(codefun2000)

题目链接 染色の树-美团2023笔试(codefun2000) 题目内容 输入描述 输出描述 输出一行一个整数表示根节点的值。 样例1 输入 3 1 1 2 2 2 输出 0 题解1 #include<bits/stdc.h> using namespace std;const int N 50005;int n, c[N];vector<int> edge[N];int dfs(…

【pytorch20】多分类问题

网络结构以及示例 该网络的输出不是一层或两层的&#xff0c;而是一个十层的代表有十分类 新建三个线性层&#xff0c;每个线性层都有w和b的tensor 首先输入维度是784&#xff0c;第一个维度是ch_out,第二个维度才是ch_in(由于后面要转置)&#xff0c;没有经过softmax函数和…

基于java+springboot+vue实现的校园外卖服务系统(文末源码+Lw)292

摘 要 传统信息的管理大部分依赖于管理人员的手工登记与管理&#xff0c;然而&#xff0c;随着近些年信息技术的迅猛发展&#xff0c;让许多比较老套的信息管理模式进行了更新迭代&#xff0c;外卖信息因为其管理内容繁杂&#xff0c;管理数量繁多导致手工进行处理不能满足广…

11.常见的Bean后置处理器

CommonAnnotationBeanPostProcessor (Resource PostConstructor PreDestroy) AutowiredAnnotationBeanPostProcessor (Autowired Value) GenericApplicationContext是一个干净的容器&#xff0c;它没有添加任何的PostProcessor处理器。 调用GenericApplicationContext.refre…