golang中defer的小坑

我们都知道,defer往往在函数的最后执行,遵循先进后出 原则,在return函数写入返回值后执行,而且传入的形参会立即读入,后续修改对其无影响

Go 函数返回过程完整流程

当一个函数执行到 return 语句时,执行顺序是

  1. 计算 return 后的表达式(如果有的话),将结果赋值给返回值;
  2. 执行所有的 defer 语句(按照后进先出 LIFO 的顺序);
  3. 函数真正返回(携带返回值)

下面来看这三个案例

go 复制代码
package main
​
import "fmt"
​
func Func1() int {
    a := 0
    defer func() {
        a++
    }()
    return a
}
​
func Func2() (a int) {
    a = 0
    defer func() {
        a++
    }()
    return
}
​
func Func3() *int {
    a := new(int)
    *a = 0
    defer func() {
        *a++
    }()
    return a
}
​
func main() {
    fmt.Println(Func1())
    fmt.Println(Func2())
    fmt.Println(*Func3())
}

输出结果为

复制代码
0 1 1

案例分析

这三个究竟有什么区别呢?

案例一:

  1. 首先,我们获得a的值0,然后会重新赋值一份给临时的返回槽(slot)当中
  2. 然后,调用defer,修改局部变量a为1,但是a已被copy一份到返回槽,所以重新赋值没有用
  3. 返回0

案例二:

  1. 首先,我们获得a的值0,然后会重新赋值一份给临时的返回槽(slot)当中
  2. 然后defer修改了引用类型a的值
  3. 返回1

案例三:

与案例一相似,但是传递的是指针,能修改地址中存储的值,所以返回的是1

相关推荐
shark_chili11 分钟前
CPU性能优化三剑客:分支预测、并行运算与超线程技术深度解析
后端
小蒜学长19 分钟前
基于Spring Boot的火灾报警系统的设计与实现(代码+数据库+LW)
java·数据库·spring boot·后端
Victor35621 分钟前
Redis(53)如何优化Redis的性能?
后端
武昌库里写JAVA21 分钟前
基于Spring Boot + Vue3的办公用品申领管理系统
java·spring boot·后端
中国lanwp22 分钟前
Spring Boot的配置文件加载顺序和规则
java·spring boot·后端
计算机毕业设计木哥2 小时前
计算机毕业设计 基于Python+Django的医疗数据分析系统
开发语言·hadoop·后端·python·spark·django·课程设计
Victor3562 小时前
Redis(52)Redis哨兵模式下如何进行版本升级?
后端
cyforkk11 小时前
Spring Boot @RestController 注解详解
java·spring boot·后端
canonical_entropy12 小时前
可逆计算:一场软件构造的世界观革命
后端·aigc·ai编程