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

本文主要是介绍【Golang】Go语言中defer与return的精妙交织:探索延迟执行与返回顺序的微妙关系,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

【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 := 1defer func() {fmt.Println("a", a) // a 1 a = 4fmt.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 := 5defer func() {t = t + 5}()return t
}
func main() {fmt.Println(f())//5
}

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

demo2

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

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

demo3

func deferFuncReturn() (result int) {i := 5defer 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 mainimport "fmt"func f() (t int) {t = 5defer 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 mainimport "fmt"func f() (t int) {t = 5defer func() {t = t + 5}()t  = 12return 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稳定的有序遍历的方式

这篇关于【Golang】Go语言中defer与return的精妙交织:探索延迟执行与返回顺序的微妙关系的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1049371

相关文章

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat

java中ssh2执行多条命令的四种方法

《java中ssh2执行多条命令的四种方法》本文主要介绍了java中ssh2执行多条命令的四种方法,包括分号分隔、管道分隔、EOF块、脚本调用,可确保环境配置生效,提升操作效率,具有一定的参考价值,感... 目录1 使用分号隔开2 使用管道符号隔开3 使用写EOF的方式4 使用脚本的方式大家平时有没有遇到自

mybatis直接执行完整sql及踩坑解决

《mybatis直接执行完整sql及踩坑解决》MyBatis可通过select标签执行动态SQL,DQL用ListLinkedHashMap接收结果,DML用int处理,注意防御SQL注入,优先使用#... 目录myBATiFBNZQs直接执行完整sql及踩坑select语句采用count、insert、u

Go语言中json操作的实现

《Go语言中json操作的实现》本文主要介绍了Go语言中的json操作的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录 一、jsOChina编程N 与 Go 类型对应关系️ 二、基本操作:编码与解码 三、结构体标签(Struc

Java 中的 equals 和 hashCode 方法关系与正确重写实践案例

《Java中的equals和hashCode方法关系与正确重写实践案例》在Java中,equals和hashCode方法是Object类的核心方法,广泛用于对象比较和哈希集合(如HashMa... 目录一、背景与需求分析1.1 equals 和 hashCode 的背景1.2 需求分析1.3 技术挑战1.4

一个Java的main方法在JVM中的执行流程示例详解

《一个Java的main方法在JVM中的执行流程示例详解》main方法是Java程序的入口点,程序从这里开始执行,:本文主要介绍一个Java的main方法在JVM中执行流程的相关资料,文中通过代码... 目录第一阶段:加载 (Loading)第二阶段:链接 (Linking)第三阶段:初始化 (Initia

python语言中的常用容器(集合)示例详解

《python语言中的常用容器(集合)示例详解》Python集合是一种无序且不重复的数据容器,它可以存储任意类型的对象,包括数字、字符串、元组等,下面:本文主要介绍python语言中常用容器(集合... 目录1.核心内置容器1. 列表2. 元组3. 集合4. 冻结集合5. 字典2.collections模块

使用Go调用第三方API的方法详解

《使用Go调用第三方API的方法详解》在现代应用开发中,调用第三方API是非常常见的场景,比如获取天气预报、翻译文本、发送短信等,Go作为一门高效并发的编程语言,拥有强大的标准库和丰富的第三方库,可以... 目录引言一、准备工作二、案例1:调用天气查询 API1. 注册并获取 API Key2. 代码实现3