golang defer的执行时机

本文详细解释了Golang中defer语句的工作原理及其执行时机。包括defer如何在函数返回、函数结束或goroutine出现panic时执行,以及defer、return与返回值之间的执行顺序。

golang中defer的执行时机


官方文档:“A “defer” statement invokes a function whose execution is deferred to the moment the surrounding function returns, either because the surrounding function executed a return statement, reached the end of its function body, or because the corresponding goroutine is panicking.”
即函数返回、函数结束或者对应的goroutine发生panic时defer执行。 

需要注意的一点是,执行return并不等于函数退出, return 语句同样可以包含一些操作,而不是单纯地返回某个值。

在golang中,函数支持多返回值,当return执行时会将各返回值压入栈中,最后再退出函数。而defer即在return执行完,函数退出前执行。所以defer执行时可修改函数返回值(不推荐这么做)。

func A() (a int) {
    defer func() {
        fmt.Println("defer...")
        a++
    }()
    a = 10
    return
}

func main() {
    fmt.Println(A())
}

这里写图片描述
defer、return、返回值的执行顺序是:return最先执行,return负责将结果写入返回值中,接着defer开始执行一些收尾工作,最后函数携带当前返回值退出。

Go 语言中,`defer` 是一种用于延迟执行函数调用的机制,通常用于资源释放、文件关闭、锁的释放等操作,以确保在函数执行结束时这些操作一定会被执行。其机制和使用方法如下: ### `defer` 的执行机制 - `defer` 调用的函数会在当前函数返回之前执行,具体是在函数内的所有 `return` 语句执行之后、函数实际返回之前执行。 - 多个 `defer` 函数的执行顺序是 **后进先出(LIFO)**,即最后声明的 `defer` 函数最先执行。 - `defer` 中的函数参数是在声明 `defer` 时就进行求值,而不是在真正执行时求值。这意味着即使变量在后续发生变化,`defer` 中使用的仍然是声明时的值[^3]。 ### `defer` 的使用方法 1. **基本使用** ```go func example() { file, _ := os.Open("test.txt") defer file.Close() // 确保在函数返回前关闭文件 // 读取文件内容 } ``` 2. **在结构体方法中使用** ```go type Student struct { Name string } func (s Student) GetName() { fmt.Println("Student name:", s.Name) } func example() { student := Student{Name: "Alice"} defer student.GetName() // 延迟调用结构体方法 // 执行其他操作 } ``` 3. **在循环中使用** - 如果在循环中使用 `defer`,需要注意变量的值在 `defer` 声明时就已经确定。为避免所有 `defer` 使用相同的变量值,可以通过将变量作为参数传递给一个新函数来捕获当前值。 ```go func example() { for i := 0; i < 3; i++ { defer func(n int) { fmt.Println("defer i:", n) }(i) } } // 输出结果为: // defer i: 2 // defer i: 1 // defer i: 0 ``` 4. **与闭包结合使用** - 如果 `defer` 使用闭包,并且闭包中引用了循环变量或其他可变变量,需要注意变量的最终值可能被所有 `defer` 共享。 ```go func example() { for i := 0; i < 3; i++ { defer func() { fmt.Println("defer i:", i) }() } } // 输出结果为: // defer i: 3 // defer i: 3 // defer i: 3 ``` ### 注意事项 - **性能问题**:在循环中频繁使用 `defer` 可能会导致性能下降,因为每次 `defer` 都需要将函数压入栈中,直到函数返回时才执行。 - **避免副作用**:由于 `defer` 的执行顺序和参数求值时机,应避免在 `defer` 中修改函数返回值或执行有副作用的操作。 - **嵌套函数中的 `defer`**:在嵌套函数中使用 `defer` 时,`defer` 只会在当前函数返回时执行,而不是在外部函数返回时执行。 ### 示例代码 ```go func DeferExample() { fmt.Println("Start") defer fmt.Println("First defer") defer fmt.Println("Second defer") fmt.Println("End") } // 输出结果为: // Start // End // Second defer // First defer ``` ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值