Golang 在编写函数闭包时会自动将变量赋值为参数吗?



这是我所指的代码:

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变量是如何工作的。每次调用函数adder0分配值时,它不应该被重置吗?

Go 以非常典型/标准的方式处理一流的函数和闭包。 有关一般关闭的一些良好背景,请参阅维基百科文章。 在这种情况下,调用adder本身:

  1. 创建名为sumint对象,其值为0
  2. 返回一个闭包:一个类似函数的thingy 1,当调用时,可以访问变量sum

adder返回的特定类似函数的东西,其调用者在普通变量中捕获,是一个接受一个参数的函数。 然后调用它,传递一个参数。 这种参数传递没有什么特别之处:它的工作方式与其他任何地方相同。 在类似函数的东西中,使用变量x得到调用者传递的值。 使用名称sum将获取捕获的int对象,无论其值如何。 从函数返回会使捕获的int仍然被捕获,因此稍后调用相同的类似函数的 thingy 会看到更新的intsum.

通过调用adder两次,你会得到两个略有不同的类似功能的东西:每个都有自己的私人sum。 这两个私有sum最初都是零。 调用您保存在pos中的类似函数的 thingy 会得到使用其中一个函数的函数。 调用略有不同的类似函数的 thingy,其值已保存在neg中,您将获得使用另一个函数的函数。

1 这个"类似函数的事物

"和实际的函数之间没有真正的区别,除了这个特定的类似函数的事物没有一个可以调用它的名称。 这或多或少就是拥有一流功能的意义。


如果您遇到可读性问题...

其原始形式是:

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上的完整代码。

当您分配posneg时,sum变量位于两个闭包中的每一个。pos闭包中的sum通过添加 1、2、3、4(斐波那契样式)来更新,而neg闭包中的sum通过在每个循环迭代中减去 2*1、2*2、2*3、2*4 来更新。

或者,更详细地说:pos := adder()分配pos一个在sum上具有闭包的函数,其中sum为 0 开始。然后,每当您调用函数pos时,它都会相应地更新sumneg和任何其他类似的赋值也是如此。

下面是一些类似的(更简单的)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

希望这有帮助。

最新更新