鍍金池/ 教程/ GO/ 6.8 閉包
4.7 strings 和 strconv 包
13.6 啟動外部命令和程序
?# 11.4 類型判斷:type-switch
12.1 讀取用戶的輸入
10.6 方法
12.2 文件讀寫
13 錯誤處理與測試
9.3 鎖和 sync 包
12.3 文件拷貝
?# 11.7 第一個(gè)例子:使用 Sorter 接口排序
?# 11.5 測試一個(gè)值是否實(shí)現(xiàn)了某個(gè)接口
6.4 defer 和追蹤
12.10 XML 數(shù)據(jù)格式
13.10 性能調(diào)試:分析并優(yōu)化 Go 程序
?# 11.1 接口是什么
2.2 Go 環(huán)境變量
2.6 安裝目錄清單
2.5 在 Windows 上安裝 Go
11.11 Printf 和反射
1.2 語言的主要特性與發(fā)展的環(huán)境和影響因素
9.0 包(package)
7.4 切片重組(reslice)
13.2 運(yùn)行時(shí)異常和 panic
10.2 使用工廠方法創(chuàng)建結(jié)構(gòu)體實(shí)例
12.8 使用接口的實(shí)際例子:fmt.Fprintf
2.4 在 Mac OS X 上安裝 Go
3.8 Go 性能說明
7.2 切片
8.0 Map
3.1 Go 開發(fā)環(huán)境的基本要求
5.6 標(biāo)簽與 goto
6.10 使用閉包調(diào)試
9.5 自定義包和可見性
4.3 常量
?# 11.2 接口嵌套接口
6.5 內(nèi)置函數(shù)
前言
10.8 垃圾回收和 SetFinalizer
2.8 Go 解釋器
13.7 Go 中的單元測試和基準(zhǔn)測試
6.8 閉包
4.9 指針
13.1 錯誤處理
10.1 結(jié)構(gòu)體定義
5.1 if-else 結(jié)構(gòu)
6.6 遞歸函數(shù)
9.9 通過 Git 打包和安裝
2.7 Go 運(yùn)行時(shí)(runtime)
10.7 類型的 String() 方法和格式化描述符
3.7 其它工具
9.6 為自定義包使用 godoc
11.12 接口與動態(tài)類型
13.3 從 panic 中恢復(fù)(Recover)
10.3 使用自定義包中的結(jié)構(gòu)體
11.14 結(jié)構(gòu)體、集合和高階函數(shù)
3.6 生成代碼文檔
9.2 regexp 包
4.1 文件名、關(guān)鍵字與標(biāo)識符
?# 11.6 使用方法集與接口
7.0 數(shù)組與切片
7.1 聲明和初始化
12.11 用 Gob 傳輸數(shù)據(jù)
5.5 Break 與 continue
1.1 起源與發(fā)展
?# 11 接口(Interfaces)與反射(reflection)
6.9 應(yīng)用閉包:將函數(shù)作為返回值
4.2 Go 程序的基本結(jié)構(gòu)和要素
8.6 將 map 的鍵值對調(diào)
6.11 計(jì)算函數(shù)執(zhí)行時(shí)間
5.0 控制結(jié)構(gòu)
10.5 匿名字段和內(nèi)嵌結(jié)構(gòu)體
4.6 字符串
3.0 編輯器、集成開發(fā)環(huán)境與其它工具
13.8 測試的具體例子
7.6 字符串、數(shù)組和切片的應(yīng)用
8.4 map 類型的切片
3.9 與其它語言進(jìn)行交互
7.3 For-range 結(jié)構(gòu)
9.7 使用 go install 安裝自定義包
6.0 函數(shù)
9.8 自定義包的目錄結(jié)構(gòu)、go install 和 go test
6.3 傳遞變長參數(shù)
13.9 用(測試數(shù)據(jù))表驅(qū)動測試
11.9 空接口
8.1 聲明、初始化和 make
6.2 函數(shù)參數(shù)與返回值
9.11 在 Go 程序中使用外部庫
3.3 調(diào)試器
4.5 基本類型和運(yùn)算符
?# 11.8 第二個(gè)例子:讀和寫
12.5 用 buffer 讀取文件
總結(jié):Go 中的面向?qū)ο?/span>
11.10 反射包
12.7 用 defer 關(guān)閉文件
9.4 精密計(jì)算和 big 包
4.4 變量
6.1 介紹
13.4 自定義包中的錯誤處理和 panicking
12.4 從命令行讀取參數(shù)
9.10 Go 的外部包和項(xiàng)目
8.3 for-range 的配套用法
3.5 格式化代碼
10.4 帶標(biāo)簽的結(jié)構(gòu)體
7.5 切片的復(fù)制與追加
?# 11.3 類型斷言:如何檢測和轉(zhuǎn)換接口變量的類型
5.4 for 結(jié)構(gòu)
4.8 時(shí)間和日期
2.3 在 Linux 上安裝 Go
12 讀寫數(shù)據(jù)
6.12 通過內(nèi)存緩存來提升性能
9.1 標(biāo)準(zhǔn)庫概述
12.6 用切片讀寫文件
10 結(jié)構(gòu)(struct)與方法(method)
8.5 map 的排序
12.9 JSON 數(shù)據(jù)格式
13.5 一種用閉包處理錯誤的模式
3.2 編輯器和集成開發(fā)環(huán)境
12.12 Go 中的密碼學(xué)
5.2 測試多返回值函數(shù)的錯誤
6.7 將函數(shù)作為參數(shù)
8.2 測試鍵值對是否存在及刪除元素
3.4 構(gòu)建并運(yùn)行 Go 程序
2.1 平臺與架構(gòu)
5.3 switch 結(jié)構(gòu)

6.8 閉包

當(dāng)我們不希望給函數(shù)起名字的時(shí)候,可以使用匿名函數(shù),例如:func(x, y int) int { return x + y }。

這樣的一個(gè)函數(shù)不能夠獨(dú)立存在(編譯器會返回錯誤:non-declaration statement outside function body),但可以被賦值于某個(gè)變量,即保存函數(shù)的地址到變量中:fplus := func(x, y int) int { return x + y },然后通過變量名對函數(shù)進(jìn)行調(diào)用:fplus(3,4)

當(dāng)然,您也可以直接對匿名函數(shù)進(jìn)行調(diào)用:func(x, y int) int { return x + y } (3, 4)

下面是一個(gè)計(jì)算從 1 到 1 百萬整數(shù)的總和的匿名函數(shù):

func() {
    sum := 0
    for i := 1; i <= 1e6; i++ {
        sum += i
    }
}()

表示參數(shù)列表的第一對括號必須緊挨著關(guān)鍵字 func,因?yàn)槟涿瘮?shù)沒有名稱。花括號 {} 涵蓋著函數(shù)體,最后的一對括號表示對該匿名函數(shù)的調(diào)用。

下面的例子展示了如何將匿名函數(shù)賦值給變量并對其進(jìn)行調(diào)用(function_literal.go):

package main

import "fmt"

func main() {
    f()
}
func f() {
    for i := 0; i < 4; i++ {
        g := func(i int) { fmt.Printf("%d ", i) } //此例子中只是為了演示匿名函數(shù)可分配不同的內(nèi)存地址,在現(xiàn)實(shí)開發(fā)中,不應(yīng)該把該部分信息放置到循環(huán)中。
        g(i)
        fmt.Printf(" - g is of type %T and has value %v\n", g, g)
    }
}

輸出:

0 - g is of type func(int) and has value 0x681a80
1 - g is of type func(int) and has value 0x681b00
2 - g is of type func(int) and has value 0x681ac0
3 - g is of type func(int) and has value 0x681400

我們可以看到變量 g 代表的是 func(int),變量的值是一個(gè)內(nèi)存地址。

所以我們實(shí)際上擁有的是一個(gè)函數(shù)值:匿名函數(shù)可以被賦值給變量并作為值使用。

練習(xí) 6.8 在 main 函數(shù)中寫一個(gè)用于打印 Hello World 字符串的匿名函數(shù)并賦值給變量 fv,然后調(diào)用該函數(shù)并打印變量 fv 的類型。

匿名函數(shù)像所有函數(shù)一樣可以接受或不接受參數(shù)。下面的例子展示了如何傳遞參數(shù)到匿名函數(shù)中:

func (u string) {
    fmt.Println(u)
    …
}(v)

請學(xué)習(xí)以下示例并思考(return_defer.go):函數(shù) f 返回時(shí),變量 ret 的值是什么?

package main

import "fmt"

func f() (ret int) {
    defer func() {
        ret++
    }()
    return 1
}
func main() {
    fmt.Println(f())
}

變量 ret 的值為 2,因?yàn)?ret++ 是在執(zhí)行 return 1 語句后發(fā)生的。

這可用于在返回語句之后修改返回的 error 時(shí)使用。

defer 語句和匿名函數(shù)

關(guān)鍵字 defer (詳見第 6.4 節(jié))經(jīng)常配合匿名函數(shù)使用,它可以用于改變函數(shù)的命名返回值。

匿名函數(shù)還可以配合 go 關(guān)鍵字來作為 goroutine 使用(詳見第 14 章和第 16.9 節(jié))。

匿名函數(shù)同樣被稱之為閉包(函數(shù)式語言的術(shù)語):它們被允許調(diào)用定義在其它環(huán)境下的變量。閉包可使得某個(gè)函數(shù)捕捉到一些外部狀態(tài),例如:函數(shù)被創(chuàng)建時(shí)的狀態(tài)。另一種表示方式為:一個(gè)閉包繼承了函數(shù)所聲明時(shí)的作用域。這種狀態(tài)(作用域內(nèi)的變量)都被共享到閉包的環(huán)境中,因此這些變量可以在閉包中被操作,直到被銷毀,詳見第 6.9 節(jié)中的示例。閉包經(jīng)常被用作包裝函數(shù):它們會預(yù)先定義好 1 個(gè)或多個(gè)參數(shù)以用于包裝,詳見下一節(jié)中的示例。另一個(gè)不錯的應(yīng)用就是使用閉包來完成更加簡潔的錯誤檢查(詳見第 16.10.2 節(jié))。

鏈接