defer与return常考点
简述 defer 和 return 返回值 的执行顺序?如果要返回一个 defer 执行后的值, 可以通过哪些方式?
大家好 我是寸铁👊
总结了一篇defer 和 return 返回值 的执行顺序探讨的文章✨
喜欢的小伙伴可以点点关注 💝
程序1
不返回具体变量
执行顺序:
进入test函数
- 先把返回的结果赋值
- 执行
defer
函数 - 如果说
defer
函数中存在打印输出的内容,则先输出defer
语句的内容。
按照指定的顺序输出,后进先出的顺序输出 - 再返回到主函数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函数
- 先把返回的结果赋值
- 执行
defer
函数 - 如果说
defer
函数中存在打印输出的内容,则先输出defer
函数语句的内容。
按照指定的顺序输出,后进先出的顺序输出 - 再返回到主函数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
,函数内部声明一个变量i
,defer
指定一个延迟函数,最后返回变量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操作后的值文章来源:https://www.toymoban.com/news/detail-808491.html
看到这里的小伙伴,恭喜你又掌握了一个知识点👊
后续有更新和变动,会在这里统一做更新,大家可以关注一波🙌
希望大家能取得胜利,坚持就是胜利💪
我是寸铁!我们下期再见💕文章来源地址https://www.toymoban.com/news/detail-808491.html
到了这里,关于【Go面试向】defer与return的执行顺序初探的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!