登录
首页 >  Golang >  Go问答

立即评估延迟调用的参数

来源:stackoverflow

时间:2024-03-30 18:36:34 135浏览 收藏

有志者,事竟成!如果你在学习Golang,那么本文《立即评估延迟调用的参数》,就很适合你!文章讲解的知识点主要包括,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~

问题内容

《go 之旅》中写道:

延迟调用的参数会立即计算,但是 在周围函数返回之前,函数调用不会被执行。

我很难理解引文的第一部分。什么叫做立即?

func def(s string) func() {
    fmt.Println("tier up")
    fmt.Println(s)
    return func(){ fmt.Println("clean up") }
}

func main() {
    defer def("defered line")()
    fmt.Println("main")
}

//Output:
//tier up
//defered line
//main
//clean up

https://play.golang.org/p/av3maexxa4r

这里推迟什么,立即评估什么?


解决方案


要了解延迟和评估的工作原理,首先让我们看一下 Spec: defer statements:

每次执行“defer”语句时,调用的函数值和参数都是 evaluated as usual 并重新保存,但实际函数不会被调用。

函数值(其调用被延迟)及其参数都会被评估。但延迟函数还没有被调用。

让我们逐步迭代您的示例:

defer f("a")

在这种情况下,函数值被评估(这将是 f),并且参数被评估,这是一个常量,所以这将是 "a"

下一步:

defer f(g("a"))

在这种情况下,将计算函数值(将是 f),并计算参数,这意味着将使用 "a" 调用 g (因为 g 的返回值是 f 的参数)。 p>

下一步:

defer f()()

如果 f 函数返回一个函数,则此值有效。函数值将被计算(这意味着 f 将被调用!),但其返回值将不会被调用,这就是将被推迟的内容。

defer f(g())()

在这种情况下,延迟函数是 f 的返回值,因此要计算延迟函数值,必须调用 f,并且为此必须先调用 gf的返回值将被延迟(不被调用)。

回到你的例子:

defer def("defered line")()

对函数值进行求值,即为def的返回值,因此调用defdef 的返回值将被延迟。它的参数被评估,但实际上它没有参数。

所以逻辑上发生的事情是这样的:

  1. 评估延迟函数值,这要求:
    • 必须调用 def 函数,这要求:
      • 评估 def 的参数,它是一个常量:“defered line”
  2. 评估延迟函数的参数;由于没有参数,因此这一步已完成。

如果我们布置上述结构,这就是按顺序发生的情况:

  • def 的参数进行求值:它是一个常量 "defered line"
  • 调用 def 并打印 tier up 及其参数:defered line
  • def 的返回值被调用,这就是延迟的。
  • 函数 main 打印:main
  • 函数 main 返回,因此现在调用延迟函数。延迟函数打印 clean up

上面的句子意味着延迟函数参数在延迟的行进行计算,但该函数将在周围函数 main 返回后运行。

延迟函数调用在周围函数返回后按后进先出顺序执行。

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

上面一行清楚地表明它将把值返回给主函数。

例如:-

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

以上函数将返回 2 值,而不是 1。这就是为什么这一行

return func(){ fmt.Println("clean up") }

将在最后调用。

有关延迟的更多信息。 Please read golang blog for defer

理论要掌握,实操不能落!以上关于《立即评估延迟调用的参数》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

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