这是我所指的代码:
package main
import "fmt"
func adder() func(int) int {
sum := 0
return func(x int) int {
sum += x
return sum
}
}
func main() {
pos, neg := adder(), adder()
for i := 0; i < 10; i++ {
fmt.Println(
pos(i),
neg(-2*i),
)
}
}
以下是运行时的输出:
0 0
1 -2
3 -6
6 -12
10 -20
15 -30
21 -42
28 -56
36 -72
45 -90
我不明白x
是如何在adder
函数的return
语句中分配的?它似乎没有在函数中的任何位置传递。 我也不明白sum
变量是如何工作的。每次调用函数adder
并0
分配值时,它不应该被重置吗?
Go 以非常典型/标准的方式处理一流的函数和闭包。 有关一般关闭的一些良好背景,请参阅维基百科文章。 在这种情况下,调用adder
本身:
- 创建名为
sum
的int
对象,其值为0
。 - 返回一个闭包:一个类似函数的thingy 1,当调用时,可以访问变量
sum
。
adder
返回的特定类似函数的东西,其调用者在普通变量中捕获,是一个接受一个参数的函数。 然后调用它,传递一个参数。 这种参数传递没有什么特别之处:它的工作方式与其他任何地方相同。 在类似函数的东西中,使用变量x
得到调用者传递的值。 使用名称sum
将获取捕获的int
对象,无论其值如何。 从函数返回会使捕获的int
仍然被捕获,因此稍后调用相同的类似函数的 thingy 会看到更新的int
sum
.
通过调用adder
两次,你会得到两个略有不同的类似功能的东西:每个都有自己的私人sum
。 这两个私有sum
最初都是零。 调用您保存在pos
中的类似函数的 thingy 会得到使用其中一个函数的函数。 调用略有不同的类似函数的 thingy,其值已保存在neg
中,您将获得使用另一个函数的函数。
"和实际的函数之间没有真正的区别,除了这个特定的类似函数的事物没有一个可以调用它的名称。 这或多或少就是拥有一流功能的意义。
如果您遇到可读性问题...
其原始形式是:
func adder() func(int) int {
sum := 0
return func(x int) int {
sum += x
return sum
}
}
让我们用一些类型名称和其他语法更改来重写它,使代码的核心保持不变。 首先,让我们做一个表示func(int) int
的名字:
type adderClosure func(int) int
然后我们可以用它来重写adder
的第一行:
func adder() adderClosure {
...
}
现在让我们在 adder 中创建一个局部变量来保存我们将要返回的函数。 为了明确和冗余,我们可以再次使用此类型:
var ret adderClosure // not good style: just for illustration
现在让我们通过这样做将该变量分配给我们的闭包:
sum := 0
ret = func(x int) int {
sum += x
return sum
}
然后我们可以return ret
返回关闭。 这是Go Playground上的完整代码。
当您分配pos
和neg
时,sum
变量位于两个闭包中的每一个。pos
闭包中的sum
通过添加 1、2、3、4(斐波那契样式)来更新,而neg
闭包中的sum
通过在每个循环迭代中减去 2*1、2*2、2*3、2*4 来更新。
或者,更详细地说:pos := adder()
分配pos
一个在sum
上具有闭包的函数,其中sum
为 0 开始。然后,每当您调用函数pos
时,它都会相应地更新sum
。neg
和任何其他类似的赋值也是如此。
下面是一些类似的(更简单的)JavaScript代码,可以在浏览器控制台中运行:
function adder() {
var sum = 0;
return function(i) {
sum += i;
return sum;
}
}
var pos = adder();
console.log( pos(1) ); // add 1 to 0 (1)
console.log( pos(2) ); // add 2 to 1 (3)
console.log( pos(3) ); // add 3 to 3 (6)
console.log( pos(4) ); // add 4 to 6 (10)
以下是有关 JavaScript 中的闭包的一些背景知识: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Closures
希望这有帮助。