鍍金池/ 教程/ GO/ 13.5 一種用閉包處理錯誤的模式
4.7 strings 和 strconv 包
13.6 啟動外部命令和程序
?# 11.4 類型判斷:type-switch
12.1 讀取用戶的輸入
10.6 方法
12.2 文件讀寫
13 錯誤處理與測試
9.3 鎖和 sync 包
12.3 文件拷貝
?# 11.7 第一個例子:使用 Sorter 接口排序
?# 11.5 測試一個值是否實(shí)現(xiàn)了某個接口
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)行時異常和 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)行時(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 計算函數(shù)執(zhí)行時間
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 第二個例子:讀和寫
12.5 用 buffer 讀取文件
總結(jié):Go 中的面向?qū)ο?/span>
11.10 反射包
12.7 用 defer 關(guān)閉文件
9.4 精密計算和 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 時間和日期
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)

13.5 一種用閉包處理錯誤的模式

每當(dāng)函數(shù)返回時,我們應(yīng)該檢查是否有錯誤發(fā)生:但是這會導(dǎo)致重復(fù)乏味的代碼。結(jié)合 defer/panic/recover 機(jī)制和閉包可以得到一個我們馬上要討論的更加優(yōu)雅的模式。不過這個模式只有當(dāng)所有的函數(shù)都是同一種簽名時可用,這樣就有相當(dāng)大的限制。一個很好的使用它的例子是 web 應(yīng)用,所有的處理函數(shù)都是下面這樣:

func handler1(w http.ResponseWriter, r *http.Request) { ... }

假設(shè)所有的函數(shù)都有這樣的簽名:

func f(a type1, b type2)

參數(shù)的數(shù)量和類型是不相關(guān)的。

我們給這個類型一個名字:

fType1 = func f(a type1, b type2)

在我們的模式中使用了兩個幫助函數(shù):

1)check:這是用來檢查是否有錯誤和 panic 發(fā)生的函數(shù):

func check(err error) { if err != nil { panic(err) } }

2)errorhandler:這是一個包裝函數(shù)。接收一個 fType1 類型的函數(shù) fn 并返回一個調(diào)用 fn 的函數(shù)。里面就包含有 defer/recover 機(jī)制,這在 13.3 節(jié)中有相應(yīng)描述。

func errorHandler(fn fType1) fType1 {
    return func(a type1, b type2) {
        defer func() {
            if err, ok := recover().(error); ok {
                log.Printf("run time panic: %v", err)
            }
        }()
        fn(a, b)
    }
}

當(dāng)錯誤發(fā)生時會 recover 并打印在日志中;除了簡單的打印,應(yīng)用也可以用 template 包(參見 15.7 節(jié))為用戶生成自定義的輸出。check() 函數(shù)會在所有的被調(diào)函數(shù)中調(diào)用,像這樣:

func f1(a type1, b type2) {
    ...
    f, _, err := // call function/method
    check(err)
    t, err := // call function/method
    check(err)
    _, err2 := // call function/method
    check(err2)
    ...
}

通過這種機(jī)制,所有的錯誤都會被 recover,并且調(diào)用函數(shù)后的錯誤檢查代碼也被簡化為調(diào)用 check(err) 即可。在這種模式下,不同的錯誤處理必須對應(yīng)不同的函數(shù)類型;它們(錯誤處理)可能被隱藏在錯誤處理包內(nèi)部。可選的更加通用的方式是用一個空接口類型的切片作為參數(shù)和返回值。

我們會在 15.5 節(jié)的 web 應(yīng)用中使用這種模式。

練習(xí)

練習(xí) 13.1recover_dividebyzero.go

用示例 13.3 中的編碼模式通過整數(shù)除以 0 觸發(fā)一個運(yùn)行時 panic。

練習(xí) 13.2panic_defer.go

閱讀下面的完整程序。不要執(zhí)行它,寫出程序的輸出結(jié)果。然后編譯執(zhí)行并驗(yàn)證你的預(yù)想。

// panic_defer.go
package main

import "fmt"

func main() {
    f()
    fmt.Println("Returned normally from f.")
}

func f() {
    defer func() {
        if r := recover(); r != nil {
            fmt.Println("Recovered in f", r)
        }
    }()
    fmt.Println("Calling g.")
    g(0)
    fmt.Println("Returned normally from g.")
}

func g(i int) {
    if i > 3 {
        fmt.Println("Panicking!")
        panic(fmt.Sprintf("%v", i))
    }
    defer fmt.Println("Defer in g", i)
    fmt.Println("Printing in g", i)
    g(i + 1)
}

輸出:

Calling g.
Printing in g 0
Printing in g 1
Printing in g 2
Printing in g 3
Panicking!
Defer in g 3
Defer in g 2
Defer in g 1
Defer in g 0
Recovered in f 4
Returned normally from f.

練習(xí) 13.3panic_defer_convint.go

寫一個 ConvertInt64ToInt 函數(shù)把 int64 值轉(zhuǎn)換為 int 值,如果發(fā)生錯誤(提示:參見 4.5.2.1 節(jié))就 panic。然后在函數(shù) IntFromInt64 中調(diào)用這個函數(shù)并 recover,返回一個整數(shù)和一個錯誤。請測試這個函數(shù)!

鏈接