在函数内部定义函数

时间:2019-05-22 18:14:49

标签: go

在定义一个使用外部作用域变量的内部函数时,我应该将变量作为参数传递给内部函数吗?

在我的示例中,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
}

2 个答案:

答案 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以便于阅读。