Go 异常处理

代码在执行的过程中可能因为一些逻辑上的问题而出现错误

func test1(a, b int) int {result := a / breturn result
}
func main() {resut := test1(10, 0)fmt.Println(resut)
}panic: runtime error: integer divide by zero          goroutine 1 [running]:                                
main.test1(...)                                       C:/Users/nlp_1/goWorkspace/src/main.go:6      
main.main()                                           C:/Users/nlp_1/goWorkspace/src/main.go:11 +0xa

error接口

Go语言引入了一个关于错误处理的标准模式,即error接口,它是Go语言内建的接口类型,该接口的定义如下:

func test1(a, b int) (result int, err error) {err = nilif b == 0 {fmt.Println("err=", err)} else {result = a / b}return
}
func main() {result, err := test1(10, 0)if err != nil {fmt.Println("err=", err)} else {fmt.Println("err=", result)}
}err= <nil>
err= 0

这种用法是非常常见的,例如,后面讲解到文件操作时,涉及到文件的打开,如下:
在这里插入图片描述
在打开文件时,如果文件不存在,或者文件在磁盘上存储的路径写错了,都会出现异常,这时可以使用error记录相应的错误信息。

panic函数

error返回的是一般性的错误,但是panic函数返回的是让程序崩溃的错误。

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

一般而言,当panic异常发生时,程序会中断运行。随后,程序崩溃并输出日志信息。日志信息包括panic value和函数调用的堆栈跟踪信息。

当然,如果直接调用内置的panic函数也会引发panic异常,panic函数接受任何值作为参数。

func test1(i int) {var arr [3]intarr[i] = 999fmt.Println(arr)
}
func main() {test1(3)
}panic: runtime error: index out of range [3] with length 3goroutine 1 [running]:                                    
main.test1(0xc000052000?)                                 C:/Users/nlp_1/goWorkspace/src/main.go:7 +0x87    
main.main()                                               C:/Users/nlp_1/goWorkspace/src/main.go:11 +0x18  

通过观察错误信息,发现确实是panic异常,导致了整个程序崩溃。

延迟调用defer

一、defer基本使用
函数定义完成后,只有调用函数才能够执行,并且一经调用立即执行。例如:

fmt.Println("hello")
fmt.Println("老王")

先输出“hello”,然后再输出“老王”。但是关键字defer⽤于延迟一个函数(或者当前所创建的匿名函数)的执行。注意,defer语句只能出现在函数的内部。

基本用法如下:

defer fmt.Println("hello")
fmt.Println("老王")

以上两行代码,输出的结果为,先输出“老王”,然后输出“hello”。

defer的应用场景:文件操作,先打开文件,执行读写操作,最后关闭文件。为了保证文件的关闭能够正确执行,可以使用defer。

2、 defer执行顺序
先看如下程序执行结果是:

defer fmt.Println("hello")
defer fmt.Println("老王")
defer fmt.Println("你好")

执行的结果是:

你好

老王

hello

总结:如果一个函数中有多个defer语句,它们会以后进先出的顺序执行。

如下程序执行的结果:

func test03(x int) {v := 100 / xfmt.Println(v)
}
func main() {defer fmt.Println("hello")defer fmt.Println("老王")defer test03(0)defer fmt.Println("你好")
}

执行结果:
你好
老王
hello
panic: runtime error: integer divide by zero

即使函数或某个延迟调用发生错误,这些调用依旧会被执⾏。

三、defer与匿名函数结合使用

我们先看以下程序的执行结果:

a := 10
b := 20
defer func() {fmt.Println("匿名函数a", a)fmt.Println("匿名函数b", b)
}()a = 100
b = 200
fmt.Println("main函数a", a)
fmt.Println("main函数b", b)

执行的结果如下:

main函数a 100
main函数b 200
匿名函数a 100
匿名函数b 200

前面讲解过,defer会延迟函数的执行,虽然立即调用了匿名函数,但是该匿名函数不会执行,等整个main()函数结束之前在去调用执行匿名函数,所以输出结果如上所示。

现在将程序做如下修改:

a := 10
b := 20
defer func(a,b int) {	//添加参数fmt.Println("匿名函数a", a)fmt.Println("匿名函数b", b)
}(a,b) //传参a = 100
b = 200
fmt.Println("main函数a", a)
fmt.Println("main函数b", b)

该程序的执行结果如下:

main函数a 100
main函数b 200
匿名函数a 10
匿名函数b 20

从执行结果上分析,由于匿名函数前面加上了defer所以,匿名函数没有立即执行。但是问题是,程序从上开始执行当执行到匿名函数时,虽然没有立即调用执行匿名函数,但是已经完成了参数的传递。

recover函数

运行时panic异常一旦被引发就会导致程序崩溃。这当然不是我们愿意看到的,因为谁也不能保证程序不会发生任何运行时错误。

Go语言为我们提供了专用于“拦截”运行时panic的内建函数——recover。它可以是当前的程序从运行时panic的状态中恢复并重新获得流程控制权。

看下面例子:

package mainimport "fmt"func testA() {fmt.Println("testA")}func testB(x int) {var a [3]inta[x] = 999
}func testC() {fmt.Println("testC")
}
func main() {testA()testB(3) //发生异常 中断程序testC()
}testA
panic: runtime error: index out of range [3] with length 3goroutine 1 [running]:                                    
main.testB(...)                                           C:/Users/nlp_1/goWorkspace/src/main.go:13         
main.main()                                               C:/Users/nlp_1/goWorkspace/src/main.go:21 +0x5b   

函数B发生了异常就不会再往下 执行了

使用recover

func testA() {fmt.Println("testA")}func testB(x int) {//设置recover()//在defer调用的函数中使用recover()defer func() {//防止程序崩溃recover()}() //匿名函数var a [3]inta[x] = 999
}func testC() {fmt.Println("testC")
}
func main() {testA()testB(3) //发生异常 中断程序testC()
}// 输出结果
testA
testC

通过以上程序,我们发现虽然TestB()函数会导致整个应用程序崩溃,但是由于在改函数中调用了recover()函数,所以整个函数并没有崩溃。虽然程序没有崩溃,但是我们也没有看到任何的提示信息,那么怎样才能够看到相应的提示信息呢?

可以直接打印recover()函数的返回结果,如下所示:

func testB(x int)  {//设置recover()//在defer调用的函数中使用recover()defer func() {//防止程序崩溃//recover()fmt.Println(recover())    //直接打印}()  //匿名函数var a [3]inta[x] = 999
}

输出结果如下:

testA
runtime error: index out of range
testC

从输出结果发现,确实打印出了相应的错误信息。

但是,如果程序没有出错,也就是数组下标没有越界,会出现什么情况呢?

func testA()  {fmt.Println("testA")}
func testB(x int)  {//设置recover()//在defer调用的函数中使用recover()defer func() {//防止程序崩溃//recover()fmt.Println(recover())}()  //匿名函数var a [3]inta[x] = 999
}func testC()  {fmt.Println("testC")
}
func main() {testA()testB(0)  //发生异常 中断程序testC()
}

输入的结果如下:

testA
<nil>
testC

这时输出的是空,但是我们希望程序没有错误的时候,不输出任何内容。

所以,程序修改如下:

func testA()  {fmt.Println("testA")
}func testB(x int)  {//设置recover()//在defer调用的函数中使用recover()defer func() {//防止程序崩溃//recover()//fmt.Println(recover())if err := recover();err != nil {fmt.Println(err)}}()  //匿名函数var a [3]inta[x] = 999
}func testC()  {fmt.Println("testC")
}
func main() {testA()testB(0)  //发生异常 中断程序testC()
}

通过以上代码,发现其实就是加了一层判断。这样就不会使得程序崩溃。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

华为云arm架构的linux系统中通过docker部署python环境

背景 有时候需要在无互联网的环境安装部署python环境,虽然可以在linux系统中直接安装python环境,但是比较复杂乱,特别是环境多的时候,其实可以通过docker打包安装的方式来实现 1、租用华为云arm加载的服务器 https://www.huaweicloud.com/product/ecs.html 2、安装doc…

请明星出席品牌周年庆活动:巧妙策略与成功之道

品牌的周年庆典是一次展示实力、感谢客户和吸引更多关注的机会。在这个特殊时刻&#xff0c;让明星出席活动演出无疑是让庆典更加引人注目和难忘的方式。明星的存在不仅能增加活动的知名度&#xff0c;还能为品牌增色不少。然而&#xff0c;邀请明星出席活动是一项复杂的任务&a…

phpcmsV9.6.0sql注入漏洞分析

目录 前言 环境准备 漏洞点 看一看parse_str函数 看一看sys_auth函数 看一看get_one函数 全局搜索sys_auth($a_k, ENCODE) 查看哪里调用了 set_cookie 查看safe_replace函数 判断登录绕过 index的业务 加载modules/wap/index.php 加载modules/attachment/attachme…

免费版Photoshop2024智能人像磨皮插件

Portraiture是一款智能磨皮插件&#xff0c;为Photoshop和Lightroom添加一键磨皮美化功能&#xff0c;快速对照片中皮肤、头发、眉毛等部位进行美化&#xff0c;无需手动调整&#xff0c;大大提高P图效率。全新4版本&#xff0c;升级AI算法&#xff0c;并独家支持多人及全身模式…

vue手写提示组件弹窗

1、弹框展示 2、message组件 新建一个message.vue <template><div class"wrapper" v-if"isShow" :class"showContent ? fadein : fadeout">{{ text }}</div> </template> <script></script> <style s…

微信怎么定时发圈?

定时发圈的妙用 在合适的时间点发布新的产品、促销活动&#xff0c;不仅能够及时提醒用户品牌的存在&#xff0c;还可以引发用户的兴趣&#xff0c;增加品牌的曝光率。 选择最佳的发朋友圈时间段&#xff0c;以确保推广内容得到最大的曝光和关注&#xff0c;提高广告投放的效果…

python元组

元组tumple tumple_python1.元组的创建2.元组是不可变序列3.元组的遍历 tumple_python -元组(tuple)是是一个有序、不可变的序列。元组用小括号 () 表示&#xff0c;其中的元素可以是任意类型&#xff0c;并且可以通过索引访问。 不可变序列与可变序列 1>不可变序列&#x…

MySQL中分区与分表的区别

MySQL中分区与分表的区别 一、分区与分表的区别 分区和分表是在处理大规模数据时的两种技术手段&#xff0c;尽管它们的目标都是提升系统的性能和数据管理的效率&#xff0c;但它们的实现方式和应用场景略有不同。 1. 分区 分区是将一个大表分割为多个更小的子表&#xff0c…

【WEB3】区块链开发入门项目之「简易NFT交易市场」

参考文章 教程英文版&#xff1a;https://docs.alchemy.com/docs/how-to-build-an-nft-marketplace-from-scratch 教程中文版&#xff1a;https://zhuanlan.zhihu.com/p/557479922?utm_id0 以太坊测试币领取&#xff1a;https://goerlifaucet.com/ 入门项目地址&#xff1…

【C++杂货铺】优先级队列的使用指南与模拟实现

文章目录 一、priority_queue的介绍二、priority_queue的使用2.1 数组中的第k个最大元素 三、priority_queue模拟实现3.1 仿函数3.2 成员变量3.3 成员函数3.3.1 构造函数3.3.2 AdjustDown3.3.3 push3.3.4 AdjustUp3.3.5 pop3.3.6 empty3.3.7 size 四、结语 一、priority_queue的…

苏州融资融券(两融)开户利率最低能做到多少?5%!

两融利率是指投资者在证券交易所通过融资买入和融券卖出股票时所支付的利息费用。两融包括融资和融券两部分&#xff0c;而融资和融券则是两种常见的金融杠杆运作方式。融资是指投资者通过向券商借入资金来买入股票。在融资中&#xff0c;投资者需要支付给券商一定的利息费用。…

面了一个测试工程师要求月薪26K,总感觉他背了很多面试题...

最近有朋友去华为面试&#xff0c;面试前后进行了20天左右&#xff0c;包含4轮电话面试、1轮笔试、1轮主管视频面试、1轮hr视频面试。 据他所说&#xff0c;80%的人都会栽在第一轮面试&#xff0c;要不是他面试前做足准备&#xff0c;估计都坚持不完后面几轮面试。 其实&…

Linux系统离线安装RabbitMQ

安装rabbitmq 1、下载安装包 首先进入官网进行安装包的下载&#xff0c;在下载时一定要注意erlong版本和rabbitmq-server版本匹配 rabbitmq版本对应关系&#xff1a;传送门 Erlong下载地址:传送门 rabbitmq-server下载地址:传送门 socat 不同版本 centos7:传送门 cent…

【Unity3D】UI Toolkit数据动态绑定

1 前言 本文将实现 cvs 表格数据与 UI Toolkit 元素的动态绑定。 如果读者对 UI Toolkit 不是太了解&#xff0c;可以参考以下内容。 UI Toolkit简介UI Toolkit容器UI Toolkit元素UI Toolkit样式选择器UI Toolkit自定义元素 本文完整资源见→UI Toolkit数据动态绑定。 2 数据…

【办公软件】微信占用C盘空间如何释放

C盘又满了&#xff0c;如果微信QQ用得多的话&#xff0c;很可能微信就占用了很多空间。当然就算安装目录选择到了其他盘&#xff0c;但是软件的缓存也有可能占用C盘。 因为微信或是其他常用软件的缓存路径都会默认保存在C盘&#xff0c;随着传输文件较多&#xff0c;每次都会自…

编码转换流

同理&#xff0c;创建f1和f2方法&#xff0c;分别测试OutputStreamWriter和InputStreamReader 也是主要分三步&#xff0c;即1创建流 2使用流 3关流 OutputStreamWriter f1方法 因为要操作流&#xff0c;所以先创建一个try-catch-finally结构&#xff0c;创建流对象Out…

ARTS 打卡 第一周,初试ARTS

前言 认识三掌柜的想必都知道&#xff0c;我持续创作技术博客已经有6年时间了&#xff0c;固定每个月发布不少于6篇博文。同时&#xff0c;自己作为一名热爱分享的开发者&#xff0c;像ARTS这样的活动自然少不了我。由于我是打算挤在一起分享&#xff0c;之前都是做了本地文档记…

在PHP8中统计数组元素个数-PHP8知识详解

在php8中&#xff0c;统计数组元素的个数&#xff0c;有下面几个函数&#xff1a;使用count()函数统计数组元素个数、使用sizeof()函数统计数组元素个数。还讲到了&#xff0c;使用array_count_values()函数来统计数组中每个元素出现的次数。 1、使用count()函数统计数组元素个…

大数据知识点之大数据5V特征

大数据的特征可以浓缩为五个英文单词&#xff0c;Volume(大量&#xff09;、Variety(多样性&#xff09;、Velocity(速度&#xff09;、Value(价值&#xff09;、Veracity(准确性&#xff09;。因为是5个特征都是以“V”开头的英文单词&#xff0c;又叫大数据5V特征。 概述&…

Redis常用应用场景

Redis是一款开源的基于内存的键值存储系统&#xff0c;它提供了多种数据结构和丰富的功能&#xff0c;适用于各种不同的应用场景。以下是Redis常用的应用场景&#xff1a; 1.缓存&#xff1a;Redis最常见的用途就是作为缓存。由于Redis存储在内存中&#xff0c;读取速度非常快…