在程序 function_return.go
中我們將會看到函數(shù) Add2 和 Adder 均會返回簽名為 func(b int) int
的函數(shù):
func Add2() (func(b int) int)
func Adder(a int) (func(b int) int)
函數(shù) Add2 不接受任何參數(shù),但函數(shù) Adder 接受一個(gè) int 類型的整數(shù)作為參數(shù)。
我們也可以將 Adder 返回的函數(shù)存到變量中(function_return.go)。
package main
import "fmt"
func main() {
// make an Add2 function, give it a name p2, and call it:
p2 := Add2()
fmt.Printf("Call Add2 for 3 gives: %v\n", p2(3))
// make a special Adder function, a gets value 2:
TwoAdder := Adder(2)
fmt.Printf("The result is: %v\n", TwoAdder(3))
}
func Add2() func(b int) int {
return func(b int) int {
return b + 2
}
}
func Adder(a int) func(b int) int {
return func(b int) int {
return a + b
}
}
輸出:
Call Add2 for 3 gives: 5
The result is: 5
下例為一個(gè)略微不同的實(shí)現(xiàn)(function_closure.go):
package main
import "fmt"
func main() {
var f = Adder()
fmt.Print(f(1), " - ")
fmt.Print(f(20), " - ")
fmt.Print(f(300))
}
func Adder() func(int) int {
var x int
return func(delta int) int {
x += delta
return x
}
}
函數(shù) Adder() 現(xiàn)在被賦值到變量 f 中(類型為 func(int) int
)。
輸出:
1 - 21 - 321
三次調(diào)用函數(shù) f 的過程中函數(shù) Adder() 中變量 delta 的值分別為:1、20 和 300。
我們可以看到,在多次調(diào)用中,變量 x 的值是被保留的,即 0 + 1 = 1
,然后 1 + 20 = 21
,最后 21 + 300 = 321
:閉包函數(shù)保存并積累其中的變量的值,不管外部函數(shù)退出與否,它都能夠繼續(xù)操作外部函數(shù)中的局部變量。
這些局部變量同樣可以是參數(shù),例如之前例子中的 Adder(as int)
。
這些例子清楚地展示了如何在 Go 語言中使用閉包。
在閉包中使用到的變量可以是在閉包函數(shù)體內(nèi)聲明的,也可以是在外部函數(shù)聲明的:
var g int
go func(i int) {
s := 0
for j := 0; j < i; j++ { s += j }
g = s
}(1000) // Passes argument 1000 to the function literal.
這樣閉包函數(shù)就能夠被應(yīng)用到整個(gè)集合的元素上,并修改它們的值。然后這些變量就可以用于表示或計(jì)算全局或平均值。
練習(xí) 6.9 不使用遞歸但使用閉包改寫第 6.6 節(jié)中的斐波那契數(shù)列程序。
練習(xí) 6.10
學(xué)習(xí)并理解以下程序的工作原理:
一個(gè)返回值為另一個(gè)函數(shù)的函數(shù)可以被稱之為工廠函數(shù),這在您需要?jiǎng)?chuàng)建一系列相似的函數(shù)的時(shí)候非常有用:書寫一個(gè)工廠函數(shù)而不是針對每種情況都書寫一個(gè)函數(shù)。下面的函數(shù)演示了如何動(dòng)態(tài)返回追加后綴的函數(shù):
func MakeAddSuffix(suffix string) func(string) string {
return func(name string) string {
if !strings.HasSuffix(name, suffix) {
return name + suffix
}
return name
}
}
現(xiàn)在,我們可以生成如下函數(shù):
addBmp := MakeAddSuffix(".bmp")
addJpeg := MakeAddSuffix(".jpeg")
然后調(diào)用它們:
addBmp("file") // returns: file.bmp
addJpeg("file") // returns: file.jpeg
可以返回其它函數(shù)的函數(shù)和接受其它函數(shù)作為參數(shù)的函數(shù)均被稱之為高階函數(shù),是函數(shù)式語言的特點(diǎn)。我們已經(jīng)在第 6.7 中得知函數(shù)也是一種值,因此很顯然 Go 語言具有一些函數(shù)式語言的特性。閉包在 Go 語言中非常常見,常用于 goroutine 和管道操作(詳見第 14.8-14.9 節(jié))。在第 11.14 節(jié)的程序中,我們將會看到 Go 語言中的函數(shù)在處理混合對象時(shí)的強(qiáng)大能力。