Go基础教程系列之defer、panic和recover详解
来源:脚本之家
时间:2023-01-07 11:46:06 446浏览 收藏
知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个Golang开发实战,手把手教大家学习《Go基础教程系列之defer、panic和recover详解》,在实现功能的过程中也带大家重新温习相关知识点,温故而知新,回头看看说不定又有不一样的感悟!
defer关键字
defer关键字可以让函数或语句延迟到函数语句块的最结尾时,即即将退出函数时执行,即便函数中途报错结束、即便已经panic()、即便函数已经return了,也都会执行defer所推迟的对象。
其实defer的本质是,当在某个函数中使用了defer关键字,则创建一个独立的defer栈帧,并将该defer语句压入栈中,同时将其使用的相关变量也拷贝到该栈帧中(显然是按值拷贝的)。因为栈是LIFO方式,所以先压栈的后执行。因为是独立的栈帧,所以即使调用者函数已经返回或报错,也一样能在它们之后进入defer栈帧去执行。
例如:
func main() { a() } func a() { println("in a") defer b() // 将b()压入defer栈中 println("leaving a") //到了这里才会执行b() } func b() { println("in b") println("leaving b") }
上面将输出:
in a leaving a in b leaving b
即便是函数已经报错,或函数已经return返回,defer的对象也会在函数退出前的最后一刻执行。
func a() TYPE{ ...CODE... defer b() ...CODE... // 函数执行出了错误 return args // 函数b()都会在这里执行 }
但注意,由于Go的作用域采用的是词法作用域,defer的定义位置决定了它推迟对象能看见的变量值,而不是推迟对象被调用时所能看见的值。
例如:
package main var x = 10 func main() { a() } func a() { println("start a:",x) // 输出10 x = 20 defer b(x) // 压栈,并按值拷贝20到栈中 x = 30 println("leaving a:",x) // 输出30 // 调用defer延迟的对象b(),输出20 } func b(x int) { println("start b:",x) }
比较下面的defer:
package main var x = 10 func main() { a() } func a() int { println("start a:", x) // 输出10 x = 20 defer func() { // 压栈,但并未传值,所以内部引用x println("in defer:", x) // 输出30 }() x = 30 println("leaving a:", x) // 输出30 return x }
上面defer推迟的匿名函数输出的值是30,它看见的不应该是20吗?先再改成下面的:
package main var x = 10 func main() { a() } func a() int { println("start a:", x) // 输出10 x = 20 defer func(x int) { println("in defer:", x) // 输出20 }(x) x = 30 println("leaving a:", x) // 输出30 return x }
这个defer推迟的对象中看见的却是20,这和第一种defer b(x)
是相同的。
原因在于defer推迟的如果是函数,它直接就在它的定义位置处评估好参数、变量。该拷贝传值的拷贝传值,该指针相见的指针相见。所以,对于第(1)和第(3)种情况,在defer的定义位置处,就将x=20拷贝给了推迟的函数参数,所以函数内部操作的一直是x的副本。而第二种情况则是直接指向它所看见的x=20那个变量,则个变量是全局变量,当执行x=30的时候会将其值修改,到执行defer推迟的对象时,它指向的x的值已经是修改过的。
再看下面这个例子,将defer放进一个语句块中,并在这个语句块中新声明一个同名变量x:
func a() int { println("start a:", x) // 输出10 x = 20 { x := 40 defer func() { println("in defer:", x) // 输出40 }() } x = 30 println("leaving a:", x) // 输出30 return x }
上面的defer定义在语句块中,它能看见的x是语句块中x=40
,它的x指向的是语句块中的x。另一方面,当语句块结束时,x=40
的x会消失,但由于defer的函数中仍有x指向40这个值,所以40这个值仍被defer的函数引用着,它直到defer执行完之后才会被GC回收。所以defer的函数在执行的时候,仍然会输出40。
如果语句块内有多个defer,则defer的对象以LIFO(last in first out)的方式执行,也就是说,先定义的defer后执行。
func main() { println("start...") defer println("1") defer println("2") defer println("3") defer println("4") println("end...") }
将输出:
start... end... 4 3 2 1
defer有什么用呢?一般用来做善后操作,例如清理垃圾、释放资源,无论是否报错都执行defer对象。另一方面,defer可以让这些善后操作的语句和开始语句放在一起,无论在可读性上还是安全性上都很有改善,毕竟写完开始语句就可以直接写defer语句,永远也不会忘记关闭、善后等操作。
例如,打开文件,关闭文件的操作写在一起:
open() defer file.Close() ... 操作文件 ...
以下是defer的一些常用场景:
- 打开关闭文件
- 锁定、释放锁
- 建立连接、释放连接
- 作为结尾输出结尾信息
- 清理垃圾(如临时文件)
panic()和recover()
panic()用于产生错误信息并终止当前的goroutine,一般将其看作是退出panic()所在函数以及退出调用panic()所在函数的函数。例如,G()中调用F(),F()中调用panic(),则F()退出,G()也退出。
注意,defer关键字推迟的对象是函数最后调用的,即使出现了panic也会调用defer推迟的对象。
例如,下面的代码中,main()中输出一个start main
之后调用a(),它会输出start a
,然后就panic了,panic()会输出panic: panic in a
,然后报错,终止程序。
func main() { println("start main") a() println("end main") } func a() { println("start a") panic("panic in a") println("end a") }
执行结果如下:
start main start a panic: panic in a goroutine 1 [running]: main.a() E:/learning/err.go:14 +0x63 main.main() E:/learning/err.go:8 +0x4c exit status 2
注意上面的end a
和end main
都没有被输出。
可以使用recover()去捕获panic()并恢复执行。recover()用于捕捉panic()错误,并返回这个错误信息。但注意,即使recover()捕获到了panic(),但调用含有panic()函数的函数(即上面的G()函数)也会退出,所以如果recover()定义在G()中,则G()中调用F()函数之后的代码都不会执行(见下面的通用格式)。
以下是比较通用的panic()和recover()的格式:
func main() { G() // 下面的代码会执行 ...CODE IN MAIN... } func G(){ defer func (){ if str := recover(); str != nil { fmt.Println(str) } }() ...CODE IN G()... // F()的调用必须在defer关键字之后 F() // 该函数内下面的代码不会执行 ...CODE IN G()... } func F() { ...CODE1... panic("error found") // 下面的代码不会执行 ...CODE IN F()... }
可以使用recover()去捕获panic()并恢复执行。但以下代码是错误的:
func main() { println("start main") a() println("end main") } func a() { println("start a") panic("panic in a") // 直接放在panic后是错误的 panic_str := recover() println(panic_str) println("end a") }
之所以错误,是因为panic()一出现就直接退出函数a()和main()了。要想recover()真正捕获panic(),需要将recover()放在defer的推迟对象中,且defer的定义必须在panic()发生之前。
例如,下面是通用格式的示例:
package main import "fmt" func main() { println("start main") b() println("end main") } func a() { println("start a") panic("panic in a") println("end a") } func b() { println("start b") defer func() { if str := recover(); str != nil { fmt.Println(str) } }() a() println("end b") }
以下是输出结果:
start main start b start a panic in a end main
注意上面的end b
、end a
都没有被输出,但是end main
输出了。
panic()是内置的函数(在包builtin中),在log
包中也有一个Panic()函数,它调用Print()输出信息后,再调用panic()。go doc log Panic
一看便知:
$ go doc log Panic func Panic(v ...interface{}) Panic is equivalent to Print() followed by a call to panic().
更多关于 Go基础教程系列之defer、panic和recover详解 请查看下面的相关链接
今天带大家了解了defer、panic、recover的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~
-
366 收藏
-
485 收藏
-
444 收藏
-
461 收藏
-
394 收藏
-
202 收藏
-
199 收藏
-
145 收藏
-
168 收藏
-
165 收藏
-
473 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习