在定义一个使用外部作用域变量的内部函数时,我应该将变量作为参数传递给内部函数吗?
在我的示例中,generate和generate2都给我相同的结果,是否有理由我应该选择其中之一?
代码选择键1来与键3、4、5组合, 然后选择键2以生成与键3、4、5的组合。
package main
import (
"fmt"
)
func main() {
fmt.Println("Hello, playground")
src := map[int][]string{
1: []string{"1", "11", "111"},
2: []string{"2", "22"},
3: []string{"3"},
4: []string{"4"},
5: []string{"5", "55"},
}
result2 := generate2(src)
fmt.Println(result2)
result := generate(src)
fmt.Println(result)
}
func generate(src map[int][]string) []string {
var combo []string
var add = func(f []string) {
for _, v := range f {
for _, p := range src[3] {
for _, q := range src[4] {
for _, r := range src[5] {
combo = append(combo, v+p+q+r)
}
}
}
}
}
add(src[1])
add(src[2])
return combo
}
func generate2(src map[int][]string) []string {
var combo []string
var add = func(f []string, combo []string, src map[int][]string) []string {
for _, v := range f {
for _, p := range src[3] {
for _, q := range src[4] {
for _, r := range src[5] {
combo = append(combo, v+p+q+r)
}
}
}
}
return combo
}
combo = add(src[1], combo, src)
combo = add(src[2], combo, src)
return combo
}
答案 0 :(得分:6)
在定义一个使用外部作用域变量的内部函数时,我应该将变量作为参数传递给内部函数吗?
这取决于您要实现的目标。
您所说的“函数内部的函数”实际上称为“闭包”(有些人称为“ lambda”)。
闭包从外部词法范围(在其主体中引用)中捕获变量。在Go中,此捕获是“按引用”或“按名称”完成的,这基本上意味着每次调用闭包时,它都会“看到”其关闭的变量的 current 值,而不是这些值变量在创建闭包时具有的时间-请观察程序:
package main
import (
"fmt"
)
func main() {
i := 42
fn := func() {
fmt.Println(i)
}
fn()
i = 12
fn()
}
将输出
42
12
相反,当您将值作为参数传递给闭包的调用时,每个调用将准确地看到传递给它的值。
我希望您现在看到选择哪种策略很大程度上取决于您想要的东西。
从概念上讲,您可能会认为闭包是特定的匿名struct
数据类型的实例,其字段是指向闭包关闭的变量的指针,并且对该闭包的每次调用都是类似于调用该类型提供的某些(匿名,唯一)方法(实际上,这就是编译器通常在背后为实现闭包所做的事情)。
这样的“方法”可能具有参数,是否应该具有参数,应该进入类型字段的内容以及该方法的参数应该是什么,可以使用常规类型使用的常用方法来判断。
答案 1 :(得分:1)
在这种情况下,两个功能之间没有功能上的区别。如您所见,局部函数可以访问局部变量,而无需显式传递它们。在您的示例中,您可能更喜欢使用generate1以便于阅读。