有志者,事竟成!如果你在学习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
,并且为此必须先调用 g
。 f
的返回值将被延迟(不被调用)。
回到你的例子:
defer def("defered line")()
对函数值进行求值,即为def
的返回值,因此调用def
。 def
的返回值将被延迟。它的参数被评估,但实际上它没有参数。
所以逻辑上发生的事情是这样的:
- 评估延迟函数值,这要求:
- 必须调用
def
函数,这要求:- 评估
def
的参数,它是一个常量:“defered line”
- 评估
- 必须调用
- 评估延迟函数的参数;由于没有参数,因此这一步已完成。
如果我们布置上述结构,这就是按顺序发生的情况:
- 对
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
理论要掌握,实操不能落!以上关于《立即评估延迟调用的参数》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注编程网公众号吧!