为什么要立即调用内联函数,而不仅仅是调用其包含的函数?

Why invoke an inline function immediately, rather than just calling its contained function?

提问人:user2417369 提问时间:9/20/2023 最后编辑:Jonathan Halluser2417369 更新时间:9/21/2023 访问量:47

问:

为结构字段赋值时。

    FlagSet: (func() *flag.FlagSet {
        fs := newFlagSet("configure")
        return fs
    })(),

我认为这等同于仅调用newFlagSet(“configure”)。这样写有什么好处。

阅读源代码时的问题。需要知道他为什么这样写。

GO 闭合

评论

0赞 Jonathan Hall 9/20/2023
我认为你是对的。这将是等效的,而且更简单。我的猜测:内联函数在过去或代码的其他部分是必要的,当它不再需要或被复制/粘贴时,它不会被重构。

答:

2赞 VonC 9/21/2023 #1

经过快速搜索,此代码来自 tailscale/tailscalecmd/tailscale/cli/configure.go#var configureCmd = &ffcli。命令{}

var configureCmd = &ffcli.Command{
    Name:      "configure",
    ShortHelp: "[ALPHA] Configure the host to enable more Tailscale features",
    LongHelp: strings.TrimSpace(`
The 'configure' set of commands are intended to provide a way to enable different
services on the host to use Tailscale in more ways.
`),
    FlagSet: (func() *flag.FlagSet {
        fs := newFlagSet("configure")
        return fs
    })(),
    Subcommands: configureSubcommands(),
    Exec: func(ctx context.Context, args []string) error {
        return flag.ErrHelp
    },
}

该代码使用函数文本(匿名函数),然后立即调用该函数。

这称为立即调用的函数表达式 (IIFE)。这在 JavaScript 等语言中更常见,但在 Go 中也很有用。

在 Go 中,IIFE 允许您隔离生成值的一段逻辑,从而为不会污染周围命名空间的变量创建作用域环境。
匿名函数中使用的变量(在本例中)不会转义到周围的代码中。这使得代码更容易推理,因为变量只在需要的时候才存在。
fs

虽然 true 等价于 ,但第二种形式的一些优点可能是:FlagSet: newFlagSet("configure"),FlagSet: (func() *flag.FlagSet { fs := newFlagSet("configure"); return fs})()

  • 可扩展性:如果将来的修改需要更复杂的操作或计算,这些更改可以很容易地合并到匿名函数中,而无需更改 的结构。newFlagSet("configure")configureCmd
  • 调试:在调试会话期间,可以很容易地注释、记录或修改封装的逻辑,而不会干扰周围的代码。

不过,从尾部代码来看,特定的 IIFE 用法似乎仅限于该实例。

评论

0赞 blackgreen 9/22/2023
有趣的是,这个解释很有说服力。不过,我并不完全相信它在任何时候都是有道理的。这似乎是可读性的重大权衡,而变量范围却是很小的优势。此外,golangbyexample 的用例是一个非常糟糕的用例,因为 IIFE 没有声明任何中间变量,该代码应该只是squareOf2 := 2*2
0赞 VonC 9/22/2023
@blackgreen我同意。这与适度;)一起使用