脚本之家

电脑版
提示:原网页已由神马搜索转码, 内容由www.jb51.net提供.
您的位置:首页脚本专栏Golang→ go defer和return执行顺序

一文详解go的defer和return的执行顺序

  更新时间:2024年07月12日 08:40:35  作者:刘小帅574 
go的defer和return是golang中的两个关键字,return用于返回函数的返回值,也可以参与一定的流程控制,defer是golang中的延迟调用,经常用于文件流的关闭,锁的解锁操作,本文给大家介绍了go的defer和return的执行顺序,需要的朋友可以参考下

详解go的defer和return的执行顺序

go的defer和return是golang中的两个关键字,return用于返回函数的返回值,也可以参与一定的流程控制,比如下面代码,return短路了后面的输出

package main
import "fmt"
// defer 和 return的详解
func main() {
foo(2)
foo(1)
}
func foo(i int) {
fmt.Println(i)
if i == 1 {
return
}
fmt.Println(i + 1)
}

结果:

2
3
1

第一次输出完整的输出了i和i+1,第二次输出被短路,只输出了1

defer是golang中的延迟调用,经常用于文件流的关闭,锁的解锁操作,defer后面的操作会在当前函数或者goroutine结束之后进行调用

package main
import "fmt"
// defer 和 return的详解
func main() {
foo()
}
func foo() {
defer fmt.Println("println defer")
fmt.Println("println foo")
}
输出:
println foo
println defer

defer自身有一些特性,比如defer和defer之间的执行顺序是先进后出,先defer的最后执行,分析下面代码:

package main
import "fmt"
// defer 和 return的详解
func main() {
foo()
}
func foo() {
defer fmt.Println("floor 3")
defer fmt.Println("floor 2")
fmt.Println("floor 1")
}
输出:
floor 1
floor 2
floor 3

根据这一特性,如果我们defer调用的代码中存在panic 的可能性,为了保证系统的运行,我们应该在前面recover而不是后面

ackage main
import "fmt"
// defer 和 return的详解
func main() {
foo()
}
func foo() {
defer func() {
panic("panic test")
}()
defer func() {
if err := recover(); err != nil {
fmt.Println("catch panic:", err)
}
}()
}
输出:
panic: panic test
package main
import "fmt"
// defer 和 return的详解
func main() {
foo()
}
func foo() {
defer func() {
if err := recover(); err != nil {
fmt.Println("catch panic:", err)
}
}()
defer func() {
panic("panic test")
}()
}
输出:
catch panic: panic test

defer和return的相互影响

defer和return的相互影响,主要是在返回值上表现,考虑下面代码,输出应该是什么:

import "fmt"
// defer 和 return的详解
func main() {
fmt.Println(foo1())
fmt.Println(foo2())
fmt.Println(foo3())
}
func foo1() int {
i := 1
defer func() { i++ }()
return i
}
func foo2() (i int) {
i = 1
defer func() { i++ }()
return i
}
func foo3() (i int) {
defer func() { i++ }()
return 1
}

输出:

1
2
2

导致上面情况的原因是

在 foo1 函数中,defer 语句中的闭包会在函数返回后执行,但是此时返回值已经确定为 1 ,所以最终返回 1 。

在 foo2 函数中,使用了命名返回值 i 。defer 语句中的闭包修改的是这个命名返回值,所以返回 2 。

在 foo3 函数中,同样使用了命名返回值 i ,defer 语句中的闭包修改了这个命名返回值,并且函数直接返回 1 ,但 defer 中的修改使得最终返回 2 。

而return的另一个特性,也会影响return和defer中代码的执行顺序

package main
import "fmt"
// defer 和 return的详解
func main() {
fmt.Println(foo1())
}
func foo1() int {
defer func() { fmt.Println("This is defer") }()
return func() int {
fmt.Println("This is return")
return 1
}()
}
输出:
This is return
This is defer
1

导致上面输出的原因是,return是非原子性的,defer会在return返回值之前执行,但return中的语句,会被全部执行,直到return锚定了某个值或者命名返回值,然后执行defer语句,最后返回return锚定的这个值

到此这篇关于一文详解go的defer和return的执行顺序的文章就介绍到这了,更多相关go defer和return执行顺序内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

    • 这篇文章主要介绍了Go语言高效I/O并发处理双缓冲和Exchanger模式实例探索,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
      2024-01-01
    • 这篇文章主要为大家详细介绍了如何利用Go语言实现分金币游戏,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
      2023-03-03
    • 本文通过代码给大家介绍了Go语言实现钉钉发送通知,代码简单易懂,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下
      2019-11-11
    • 今天小编就为大家分享一篇关于在go中使用omitempty的代码实例,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
      2019-04-04
    • 这篇文章主要介绍了如何使用Go来实现优雅重启服务,本文给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
      2019-11-11
    • 这篇文章主要介绍了goland等待锁问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧
      2020-11-11
    • 这篇文章主要介绍了sublime安装支持go和html的插件,需要的朋友可以参考下
      2015-01-01
    • 这篇文章主要介绍了Golang算法之田忌赛马问题实现方法,结合具体实例形式分析了基于Go语言的田忌赛马问题原理与算法实现技巧,需要的朋友可以参考下
      2017-02-02
    • 这篇文章主要为大家介绍了Go语言fsnotify接口实现监测文件修改的示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
      2022-06-06
    • 在 Go 语言中,我们常用的数据结构有在Go语言中,你可以初始化不同的数据结构,例如数组、切片、结构体、指针、map等,本文将给大家介绍一下Golang变量直接初始化的方法,需要的朋友可以参考下
      2023-08-08

    最新评论