登录
首页 >  Golang >  Go问答

我为什么在以下带有 defer 的 Golang 代码示例中获得了0和1

来源:stackoverflow

时间:2024-03-24 21:36:42 378浏览 收藏

在 Golang 中,带有 `defer` 语句的代码在不同情况下产生不同的结果。对于传入参数,`defer` 在返回值计算完成后执行,对返回值无影响。然而,对于结果参数,`defer` 可以修改返回变量,影响实际返回值。

问题内容

对于以两种不同方式声明的变量,调用 defer 会产生不同的结果

package main

import (
    "fmt"
)

func c(i int) int {
    defer func() { i++ }()
    return i
}

func c1() (i int) {
    defer func() { i++ }()
    return i
}

func c2() (i int) {
    defer func() { i++ }()
    return 2
}

func main() {
    fmt.Println(c(0)) // Prints 0
    fmt.Println(c1()) // Prints 1
    fmt.Println(c2()) // Prints 3 Thank you icza
}

https://play.golang.org/p/gfnncz--dkh


解决方案


在第一个示例中,i 是一个(传入)参数。在 return 语句处,计算返回值,延迟函数在此之后运行,并且在其中递增 i 对返回值没有影响。

在第二个示例中,i 是结果参数的名称。在 return 语句中,您显式返回值 i,然后将该值分配给返回值 i (这是一个无操作)。但是延迟函数可以修改返回“变量”的值,如果这样做,将会对实际返回值产生影响。

如果我们添加另一个示例,这一点就会变得更清楚:

func c2() (i int) {
    defer func() { i++ }()
    return 2
}

此函数将返回 3,因为 return 2 语句会将 2 分配给 i,然后延迟函数将递增此值,因此返回值为 3。在 Go Playground 上尝试这个。Spec: Return statements: 中的相关部分

在执行任何延迟函数之前指定结果集的“return”语句设置结果参数。

一般来说,如果函数(或方法)具有命名结果参数,则返回值将始终是这些变量的值,但一定不要忘记 return 语句可能会为这些结果参数分配新值,并且它们可能会在 return 语句之后由延迟函数修改。

Spec: Defer statements中提到了这一点:

例如,如果延迟函数是 function literal,并且周围函数具有在文字范围内的 named result parameters,则延迟函数可以在返回结果参数之前访问和修改结果参数。

博文Defer, Panic and Recover中也提到了这一点:

延迟函数可以读取并分配给返回函数的命名返回值。

还有Effective Go: Recover

如果 doparse 发生紧急情况,恢复块会将返回值设置为 nil — 延迟函数可以修改命名返回值。

查看相关问题:How to return a value in a Go function that panics?

终于介绍完啦!小伙伴们,这篇关于《我为什么在以下带有 defer 的 Golang 代码示例中获得了0和1》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>