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.hqwc.cn/news/108207.html

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

相关文章

2023年华数杯数学建模A题隔热材料的结构优化控制研究解题全过程文档及程序

2023年华数杯全国大学生数学建模 A题 隔热材料的结构优化控制研究 原题再现&#xff1a; 新型隔热材料 A 具有优良的隔热特性&#xff0c;在航天、军工、石化、建筑、交通等高科技领域中有着广泛的应用。   目前&#xff0c;由单根隔热材料 A 纤维编织成的织物&#xff0c;…

全能文字转语音 - 具备真人发声效果的智能配音软件

现在的配音软件软件很多&#xff0c;各种类型的都比较多&#xff0c;这就来给你分享几款好用的配音软件。不论是制作短视频还是制作平常音频都可以用上&#xff0c;效果直逼真人&#xff01; 一、&#xff1a;悦音配音 悦音AI配音&#xff0c;媲美真人的AI配音技术&#xff0c…

socket编程|TCP

一.套接字概念 套接字&#xff08;Socket&#xff09;是一种用于网络通信的编程接口&#xff0c;它提供了一种机制&#xff0c;使得不同计算机上的应用程序能够通过网络进行通信和交换数据。 套接字可以看作是应用程序和网络之间的端点&#xff0c;它定义了应用程序与网络之间…

maven运行报错解决

在IDEA上运行较大项目时&#xff0c;编译量很大&#xff0c;可能会报出 Error:java: java.lang.OutOfMemoryError: Java heap space 的错误&#xff0c;解决方法如下&#xff1a; java.lang.OutOfMemoryError是内存不足导致的&#xff0c;因此需要修改Idea运行项目的内存大小。…

html给下拉框添加搜索、分页功能(通过ajax从服务器获取搜索数据)

文章目录 下拉框搜索分页功能开发功能使用源码和Demo&#xff08;点个赞再走咯&#xff09;test.htmlsearchable-select.csssearchserver-select.js 下拉框搜索分页功能开发 最近需要开发一个下拉框从服务器通过Ajax请求搜索数据库并且分页的组件&#xff0c;源码和demo放在下面…

数据结构——二叉树

堆的简易建立 前言堆的顺序表实现一、Heap.h头文件向下调整算法复杂度向上调整算法复杂度 二、Heap.c功能函数文件三、Test.c测试函数文件四、运行结果展示五、完整代码展示 二叉树的链表实现回顾二叉树的概念链式二叉树代码模拟二叉树的遍历 总结 前言 现在我们开始学习堆的建…

C语言入门Day_19 初识函数

目录 1.函数的定义 2.函数的调用 3.易错点 4.思维导图 前言&#xff1a; printf()我们已经很熟悉了&#xff0c;它有一个特定的功能&#xff0c;就是在屏幕上输出一行文字。之前的课程我们都称呼printf()为一个功能&#xff0c;实际上ta在编程中有个特定的名字——函数。 …

获取Windows 10中的照片(旧版)下载

Windows 10中的新版照片应用&#xff0c;目前发现无法直接打开部分iOS设备上存储的照片。需要使用照片&#xff08;旧版&#xff09;才行。 但目前应用商店中无法直接搜索到照片&#xff08;旧版&#xff09;&#xff0c;因此笔者提供如下链接&#xff0c;可以直接访问并呼出W…

C++零碎记录(十三)

23. 多态 23.1 多态简介 ① 多态是C面向对象三大特性之一。 ② 多态分为两类&#xff1a; 1. 静态多态&#xff1a;函数重载和运算符重载属于静态多态&#xff0c;复用函数名。 2. 动态多态&#xff1a;派生类和 虚函数实现运行时多态。 ③ 静态多态和动态多态区别&#xff…

通过 chatgpt 协助完成网站数据破解

Chatgpt 的出现极大地提升了程序员的工作效率&#xff0c;常见的使用场景包括代码自动生成、代码静态检查等&#xff0c;那么 chatgpt 能否用于某些网站的数据破解工作呢&#xff1f; 问题 某天线上服务开始报警&#xff0c;原来是某个视频网站无法获取到其 cdn 地址导致的下…

K8s上安装gitlab-ce

文章目录 K8s上安装gitlab-ce操作如下gitlab-deployment.yml K8s上安装gitlab-ce 前言   使用pv-pvc来持久化gitlab的数据&#xff0c;配置&#xff0c;日志文件。   pod启动后需要需要修改external_url然后重启pod。 操作如下 mkdir -p /mnt/data01/gitlab ctr -n k8s.…

内外统一的边缘原生云基础设施架构——火山引擎边缘云

近日&#xff0c;火山引擎边缘云边缘计算架构师郭少巍在LiveVideoStack Con 2023上海站围绕火山引擎边缘云海量分布式节点和上百T带宽&#xff0c;结合边缘计算在云基础设施架构方面带来的挑战&#xff0c;分享了面对海量数据新的应用形态对低时延和分布式架构的需求&#xff0…