见代码:
package main import ( "fmt" ) func main() { var whatever [5]struct{} for i := range whatever { fmt.Println(i) } //part 1 for i := range whatever { defer func() { fmt.Println(i)}() } //part 2 for i := range whatever { defer func(n int) { fmt.Println(n)}(i) } //part 3 }part 1不必解释什么, 一目了然, 对于part 2,这是典型的一个go语言闭包, 它捕获了变量"i", 注意是持有的”引用”不是副本哟! 所以当part2 的for循环结束时, 变量“i”的值正好为4,所以,闭包的输出皆为4。 对于part 3 ,虽然也是闭包, 但是没有捕获变量“i”, 而是通过普通的传参方式调用闭包, 并且, 对于形如: defer f(e) 的语句, f函数或闭包虽然不会立即执行, 但是其参数表达式会被立即评估,就是求值, 形如:f(0), f(1) ... f(4) ; 并且由于defer调用是以先进后出的方式执行的, 所以part3 的输出为: 4,3,2,1,0 ;所以part2在part3之后执行,故输出颠倒!
原文: http://stackoverflow.com/questions/16010694/how-golangs-defer-capture-closures-parameter
注意: 此文章只是我个人笔记, 如有错漏,请一定指正, 共同学习, 我的邮箱: htyu_0203_39@sina.com
有疑问加站长微信联系(非本文作者)