一尘不染

为什么在Golang的闭包主体之后添加“()”?

go

我正在阅读The Go Programming Language Specifications,发现在关闭正文之后,自己对“()”并不真正了解:

Function literals

func(ch chan int){ch < -ACK } (replyChan) `

Defer statements的示例中:

// f returns 1
func f() (result int) {
    defer func() {
        result++
    }() // why and how?
    return 0
}

我不清楚关闭主体后添加和使用“()”的原因,希望有人可以清楚地解释一下。


阅读 237

收藏
2020-07-02

共1个答案

一尘不染

这并不是说()必须(仅)一个后添加
deferdefer语句的语言规范要求其“
Expression” 始终 必须是函数调用。

为什么会这样呢?不管是否“推迟”,它与任何其他功能相同:

考虑:

func f() int { return 42 }

a := f

b := f()

第一表达式RHS是函数值。在第二个版本中,RHS是函数 返回 的值-即函数调用。

的语义也是如此:

defer f

defer f()

除了在“ defer”的上下文中第一个版本没有意义外,因此规范中提到它必须是第二种形式(仅)。

由于与上面讨论的’defer’语句之外的函数调用具有正交性,因此IMHO也更容易学习。

还要注意,函数调用不仅是fn-expr后跟(),而且表达式列表通常在括号内(包括一个空列表)。之间有很大的区别:

for i := range whatever {
        defer func() { fmt. Println(i) }()
}

for i := range whatever {
        defer func(n int) { fmt. Println(n) }(i)
}

关闭时,第一个版本在当下打印的“我”的值 执行 ,第二打印“我”在当下的价值时,defer语句 执行。

2020-07-02