Golang 中什么时候在匿名函数中声明变量?

问题描述 投票:0回答:1

我在代码中使用了一些匿名函数,我试图理解函数中调用的这两个代码片段之间的区别(如果有的话):

defer func(s *Service, ID string) {
    err := s.Deprovision(ID)
    if err != nil {
        s.logger.Error(err)
    }
}(service, identifier)


defer func() {
    err := service.Deprovision(identifier)
    if err != nil {
        service.logger.Error(err)
    }
}()

一种直接使用父函数中可用的变量,另一种本质上是在匿名函数中声明这些变量,然后指定要传入的父变量。

这有什么不同吗?如果是这样,我什么时候需要注意这一点?

function go closures anonymous
1个回答
0
投票

您提供的两个代码片段都使用带有

defer
语句的匿名函数。然而,它们的不同之处在于如何捕获和使用封闭函数中的变量。让我们分解差异并讨论何时需要注意它们:

  1. 直接使用变量(第一个片段): 在第一个代码片段中,匿名函数直接使用封闭函数作用域中的参数

    s
    ID
    。这意味着这些变量被闭包捕获,并且可以在匿名函数中直接访问,而无需作为参数传递。

  2. 将变量作为参数传递(第二个片段): 在第二个代码片段中,匿名函数不使用封闭函数作用域中的参数。相反,它从封闭函数的作用域中按值捕获

    service
    identifier
    变量,然后在函数体内使用它们。这本质上相当于在匿名函数中声明和初始化新的局部变量。

这有什么不同吗?

是的,这两种方法之间存在差异,它们之间的选择取决于您的具体用例。

  • 范围和清晰度: 第一种方法更明确,对于阅读代码的人来说可能更清楚,因为它准确地显示了封闭范围中正在使用哪些变量。这可以帮助防止潜在的错误并提高代码的可读性。

  • 性能和内存: 第二种方法(通过值捕获变量)可能会对性能和内存使用产生很小的影响,特别是当封闭函数的作用域包含大型对象时。这是因为每个闭包都会有自己的捕获变量副本,这可能会增加内存消耗。

何时要保持正念:

  1. 变量更改:如果变量的值(在您的情况下是

    service
    identifier
    )在定义匿名函数之后但在执行之前(由于循环或其他控制结构)发生变化,则第二种方法可能会出乎意料。

  2. 闭包和 Goroutines: 当将 goroutine 与闭包结合使用时,捕获的变量会对并发行为产生微妙的影响。确保您了解闭包如何捕获变量以及它们如何影响并发执行。

  3. 性能注意事项:如果内存使用是一个问题或者按值捕获会导致性能问题,您可能需要考虑第一种方法,它直接使用现有变量。

在大多数情况下,两种方法之间的差异可能并不显着,但在做出选择时最好了解这些注意事项。选择最适合您的特定用例并提高代码清晰度和可维护性的方法。

© www.soinside.com 2019 - 2024. All rights reserved.