【Golang】Go语言中defer与return的精妙交织:探索延迟执行与返回顺序的微妙关系
创始人
2025-01-16 08:04:38
0

【Golang】Go语言中defer与return的精妙交织:探索延迟执行与返回顺序的微妙关系

大家好 我是寸铁👊
总结了一篇defer 和 return 返回值 的执行顺序探讨的文章✨
喜欢的小伙伴可以点点关注 💝

在这里插入图片描述


前言

在Go语言中,defer return是两个非常重要的关键字,它们在函数执行过程中起着至关重要的作用。defer 用于延迟执行一个函数调用,通常用于在函数执行结束后进行一些清理工作或资源释放操作。而return则用于从函数中返回结果并终止函数的执行。在Go语言中,defer 语句的执行时机和返回值的处理方式涉及到一些微妙的规则,特别是在 defer 和 return 同时出现时,其执行顺序可能会引发一些混淆和困惑
本文将深入探讨Go语言中 deferreturn的执行顺序,并解释它们之间的关系。我们将详细讨论在不同情况下,deferreturn 的执行顺序及其影响,以帮助读者更好地理解和使用这两个关键字,从而写出更加健壮、清晰的代码。

defer与return常考点

简述 defer return 返回值 的执行顺序?如果要返回一个 defer 执行后的值, 可以通过哪些方式?


程序1

不返回具体变量

执行顺序:
进入test函数

  1. 先把返回的结果赋值
  2. 执行defer函数
  3. 如果说defer函数中存在打印输出的内容,则先输出defer语句的内容。
    按照指定的顺序输出,后进先出的顺序输出
  4. 再返回到主函数main中,把之前赋值的返回的结果输出来

注意:return的结果是不会受defer函数而影响的,但是假如在defer外return之前进行赋值
则会返回当前更新的值。

demo

 func Test() int {     a := 1     defer func() {         fmt.Println("a", a) // a 1          a = 4         fmt.Println("2", a) // 2 4     }()     return a // 1 } func main() {     fmt.Println(Test()) //1  } 

程序2

返回指定的变量

执行顺序:
进入test函数

  1. 先把返回的结果赋值
  2. 执行defer函数
  3. 如果说defer函数中存在打印输出的内容,则先输出defer函数语句的内容。
    按照指定的顺序输出,后进先出的顺序输出
  4. 再返回到主函数main中,把之前赋值的返回的结果输出来

注意:函数中先对返回的结果进行定义:= 而不是直接赋值=,return的结果是不会受defer函数而影响的,但是假如在defer外 return之前进行赋值 则会返回当前更新的值。如下demo1和demo2


函数内先对返回值定义:=

demo1

  func f() (r int) {     t := 5     defer func() {         t = t + 5     }()     return t } func main() {     fmt.Println(f())//5 } 

验证一下return的结果会不会受到defer函数的影响
根据输出结果,是不会受到影响的!

demo2

func f() (r int) {     t := 5     defer func() {         t = t + 5         t = 12     }()     t = 20     return t //此时t从5 --> 20 } func main() {     fmt.Println(f())//20 }  

再来看这种情况,与上面的demo不同的点在于返回参数result在defer中参与了运算

demo3

func deferFuncReturn() (result int) {     i := 5     defer func() {        result += 5     }()     return i //这里由于返回的result参与了运算,所以会经过defer处理,返回的结果为10 } 

函数说明:

函数拥有一个具名返回值result,函数内部声明一个变量idefer指定一个延迟函数,最后返回变量i。延迟函数中增加result

参考答案:

函数输出10。函数的return语句并不是原子的,实际执行分为设置返回值—>ret,defer语句实际执行在返回前,即拥有defer的函数返回过程是:设置返回值—>执行defer—>ret。所以return语句先把result设置为i的值,即5,defer语句中又把result递增5,所以最终返回10。

小结:
所以,这里的关键点在于具名返回值(也就是返回的参数列表中的变量)是否参与defer 运算,参与运算则返回值会受到defer的处理影响,不参与则与之无关。


函数内没有对返回值进行定义

函数中没有对返回值进行定义(:=) 函数返回值先进行初始化 之后在函数内进行赋值 之后defer函数可以返回经过defer操作后的值

返回值是与defer中的程序有关
demo

package main  import "fmt"  func f() (t int) {     t = 5     defer func() {        t = t + 5     }()     return t //返回值是与defer中的程序有关  } func main() {     fmt.Println(f()) } 

分析:defer在return之后执行,但是在函数退出之前,defer可以修改返回值。这里是先对返回值赋值为5 之后待返回值赋值后 执行defer函数 对t的值进行修改 之后返回的值t也会被defer函数所修改。


再来看一下这一组demo

返回值是与defer中的程序有关 如果在defer的外面重新给t赋值后 则返回的结果值会随之修改

demo

 package main  import "fmt"  func f() (t int) {     t = 5     defer func() {        t = t + 5     }()     t  = 12     return t //返回值是与defer中的程序有关 如果在defer的外面重新给t赋值后 则返回的结果值会随之修改 } func main() {     fmt.Println(f())//17 } 

分析:这里先对返回值t进行赋值 此时t为5 此时还没执行defer函数 是在赋值给return 的结果完毕后才执行defer函数 很明显 这里t又被修改为12 此时t的值赋值完毕 开始执行defer函数,对t的值进行修改,此时t = 12 + 5 = 17 返回给主函数 输出17

返回一个defer执行后的值

如果要返回一个 defer 执行后的值, 可以通过那些方式?
见上:函数中没有对返回值进行定义的情况

简述:函数中没有对返回值进行定义(:=) 函数返回值先进行初始化 之后在函数内进行赋值 之后defer函数可以返回经过defer操作后的值


结语

通过本文的讨论,我们更深入地理解了Go语言中 defer 和 return 的执行顺序以及它们之间的关系。defer 语句的延迟执行在函数返回前确保了一些清理工作的完成,而 return 的执行顺序则会受到 defer 的影响,需要注意避免出现意外情况。在编写Go语言程序时,合理使用 defer 和 return 可以使代码更加清晰、简洁,并保证资源的正确释放和程序逻辑的正确执行。希望本文能够帮助读者更好地理解和应用这两个关键字,从而写出高质量的Go语言代码。


看到这里的小伙伴,恭喜你又掌握了一个知识点👊
后续有更新和变动,会在这里统一做更新,大家可以关注一波🙌
希望大家能取得胜利,坚持就是胜利💪
我是寸铁!我们下期再见💕


往期好文💕

保姆级教程

【保姆级教程】Windows11下go-zero的etcd安装与初步使用

【保姆级教程】Windows11安装go-zero代码生成工具goctl、protoc、go-zero

【Go-Zero】手把手带你在goland中创建api文件并设置高亮


报错解决

【Go-Zero】Error: user.api 27:9 syntax error: expected ‘:‘ | ‘IDENT‘ | ‘INT‘, got ‘(‘ 报错解决方案及api路由注意事项

【Go-Zero】Error: only one service expected goctl一键转换生成rpc服务错误解决方案

【Go-Zero】【error】 failed to initialize database, got error Error 1045 (28000):报错解决方案

【Go-Zero】Error 1045 (28000): Access denied for user ‘root‘@‘localhost‘ (using password: YES)报错解决方案

【Go-Zero】type mismatch for field “Auth.AccessSecret“, expect “string“, actual “number“报错解决方案

【Go-Zero】Error: user.api 30:2 syntax error: expected ‘)‘ | ‘KEY‘, got ‘IDENT‘报错解决方案

【Go-Zero】Windows启动rpc服务报错panic:context deadline exceeded解决方案


Go面试向

【Go面试向】defer与time.sleep初探

【Go面试向】defer与return的执行顺序初探

【Go面试向】Go程序的执行顺序

【Go面试向】rune和byte类型的认识与使用

【Go面试向】实现map稳定的有序遍历的方式

相关内容

热门资讯

一分钟内幕!科乐吉林麻将系统发... 一分钟内幕!科乐吉林麻将系统发牌规律,福建大玩家确实真的是有挂,技巧教程(有挂ai代打);所有人都在...
一分钟揭秘!微扑克辅助软件(透... 一分钟揭秘!微扑克辅助软件(透视辅助)确实是有挂(2024已更新)(哔哩哔哩);1、用户打开应用后不...
五分钟发现!广东雀神麻雀怎么赢... 五分钟发现!广东雀神麻雀怎么赢,朋朋棋牌都是是真的有挂,高科技教程(有挂方法)1、广东雀神麻雀怎么赢...
每日必看!人皇大厅吗(透明挂)... 每日必看!人皇大厅吗(透明挂)好像存在有挂(2026已更新)(哔哩哔哩);人皇大厅吗辅助器中分为三种...
重大科普!新华棋牌有挂吗(透视... 重大科普!新华棋牌有挂吗(透视)一直是有挂(2021已更新)(哔哩哔哩)1、完成新华棋牌有挂吗的残局...
二分钟内幕!微信小程序途游辅助... 二分钟内幕!微信小程序途游辅助器,掌中乐游戏中心其实存在有挂,微扑克教程(有挂规律)二分钟内幕!微信...
科技揭秘!jj斗地主系统控牌吗... 科技揭秘!jj斗地主系统控牌吗(透视)本来真的是有挂(2025已更新)(哔哩哔哩)1、科技揭秘!jj...
1分钟普及!哈灵麻将攻略小,微... 1分钟普及!哈灵麻将攻略小,微信小程序十三张好像存在有挂,规律教程(有挂技巧)哈灵麻将攻略小是一种具...
9分钟教程!科乐麻将有挂吗,传... 9分钟教程!科乐麻将有挂吗,传送屋高防版辅助(总是存在有挂)1、完成传送屋高防版辅助透视辅助安装,帮...
每日必看教程!兴动游戏辅助器下... 每日必看教程!兴动游戏辅助器下载(辅助)真是真的有挂(2025已更新)(哔哩哔哩)1、打开软件启动之...