鍍金池/ 教程/ GO/ 5.2 測(cè)試多返回值函數(shù)的錯(cuò)誤
4.7 strings 和 strconv 包
13.6 啟動(dòng)外部命令和程序
?# 11.4 類型判斷:type-switch
12.1 讀取用戶的輸入
10.6 方法
12.2 文件讀寫
13 錯(cuò)誤處理與測(cè)試
9.3 鎖和 sync 包
12.3 文件拷貝
?# 11.7 第一個(gè)例子:使用 Sorter 接口排序
?# 11.5 測(cè)試一個(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 語(yǔ)言的主要特性與發(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 性能說(shuō)明
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 中的單元測(cè)試和基準(zhǔn)測(cè)試
6.8 閉包
4.9 指針
13.1 錯(cuò)誤處理
10.1 結(jié)構(gòu)體定義
5.1 if-else 結(jié)構(gòu)
6.6 遞歸函數(shù)
9.9 通過(guò) Git 打包和安裝
2.7 Go 運(yùn)行時(shí)(runtime)
10.7 類型的 String() 方法和格式化描述符
3.7 其它工具
9.6 為自定義包使用 godoc
11.12 接口與動(dòng)態(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)識(shí)符
?# 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 的鍵值對(duì)調(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 測(cè)試的具體例子
7.6 字符串、數(shù)組和切片的應(yīng)用
8.4 map 類型的切片
3.9 與其它語(yǔ)言進(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 傳遞變長(zhǎng)參數(shù)
13.9 用(測(cè)試數(shù)據(jù))表驅(qū)動(dòng)測(cè)試
11.9 空接口
8.1 聲明、初始化和 make
6.2 函數(shù)參數(shù)與返回值
9.11 在 Go 程序中使用外部庫(kù)
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 自定義包中的錯(cuò)誤處理和 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 類型斷言:如何檢測(cè)和轉(zhuǎn)換接口變量的類型
5.4 for 結(jié)構(gòu)
4.8 時(shí)間和日期
2.3 在 Linux 上安裝 Go
12 讀寫數(shù)據(jù)
6.12 通過(guò)內(nèi)存緩存來(lái)提升性能
9.1 標(biāo)準(zhǔn)庫(kù)概述
12.6 用切片讀寫文件
10 結(jié)構(gòu)(struct)與方法(method)
8.5 map 的排序
12.9 JSON 數(shù)據(jù)格式
13.5 一種用閉包處理錯(cuò)誤的模式
3.2 編輯器和集成開發(fā)環(huán)境
12.12 Go 中的密碼學(xué)
5.2 測(cè)試多返回值函數(shù)的錯(cuò)誤
6.7 將函數(shù)作為參數(shù)
8.2 測(cè)試鍵值對(duì)是否存在及刪除元素
3.4 構(gòu)建并運(yùn)行 Go 程序
2.1 平臺(tái)與架構(gòu)
5.3 switch 結(jié)構(gòu)

5.2 測(cè)試多返回值函數(shù)的錯(cuò)誤

Go 語(yǔ)言的函數(shù)經(jīng)常使用兩個(gè)返回值來(lái)表示執(zhí)行是否成功:返回某個(gè)值以及 true 表示成功;返回零值(或 nil)和 false 表示失?。ǖ?4.4 節(jié))。當(dāng)不使用 true 或 false 的時(shí)候,也可以使用一個(gè) error 類型的變量來(lái)代替作為第二個(gè)返回值:成功執(zhí)行的話,error 的值為 nil,否則就會(huì)包含相應(yīng)的錯(cuò)誤信息(Go 語(yǔ)言中的錯(cuò)誤類型為 error: var err error,我們將會(huì)在第 13 章進(jìn)行更多地討論)。這樣一來(lái),就很明顯需要用一個(gè) if 語(yǔ)句來(lái)測(cè)試執(zhí)行結(jié)果;由于其符號(hào)的原因,這樣的形式又稱之為 comma,ok 模式(pattern)。

在第 4.7 節(jié)的程序 string_conversion.go 中,函數(shù) strconv.Atoi 的作用是將一個(gè)字符串轉(zhuǎn)換為一個(gè)整數(shù)。之前我們忽略了相關(guān)的錯(cuò)誤檢查:

anInt, _ = strconv.Atoi(origStr)

如果 origStr 不能被轉(zhuǎn)換為整數(shù),anInt 的值會(huì)變成 0 而 _ 無(wú)視了錯(cuò)誤,程序會(huì)繼續(xù)運(yùn)行。

這樣做是非常不好的:程序應(yīng)該在最接近的位置檢查所有相關(guān)的錯(cuò)誤,至少需要暗示用戶有錯(cuò)誤發(fā)生并對(duì)函數(shù)進(jìn)行返回,甚至中斷程序。

我們?cè)诘诙€(gè)版本中對(duì)代碼進(jìn)行了改進(jìn):

示例 1:

示例 5.3 string_conversion2.go

package main

import (
    "fmt"
    "strconv"
)

func main() {
    var orig string = "ABC"
    // var an int
    var newS string
    // var err error

    fmt.Printf("The size of ints is: %d\n", strconv.IntSize)      
    // anInt, err = strconv.Atoi(origStr)
    an, err := strconv.Atoi(orig)
    if err != nil {
        fmt.Printf("orig %s is not an integer - exiting with error\n", orig)
        return
    } 
    fmt.Printf("The integer is %d\n", an)
    an = an + 5
    newS = strconv.Itoa(an)
    fmt.Printf("The new string is: %s\n", newS)
}

這是測(cè)試 err 變量是否包含一個(gè)真正的錯(cuò)誤(if err != nil)的習(xí)慣用法。如果確實(shí)存在錯(cuò)誤,則會(huì)打印相應(yīng)的錯(cuò)誤信息然后通過(guò) return 提前結(jié)束函數(shù)的執(zhí)行。我們還可以使用攜帶返回值的 return 形式,例如 return err。這樣一來(lái),函數(shù)的調(diào)用者就可以檢查函數(shù)執(zhí)行過(guò)程中是否存在錯(cuò)誤了。

習(xí)慣用法

value, err := pack1.Function1(param1)
if err != nil {
    fmt.Printf("An error occured in pack1.Function1 with parameter %v", param1)
    return err
}
// 未發(fā)生錯(cuò)誤,繼續(xù)執(zhí)行:

由于本例的函數(shù)調(diào)用者屬于 main 函數(shù),所以程序會(huì)直接停止運(yùn)行。

如果我們想要在錯(cuò)誤發(fā)生的同時(shí)終止程序的運(yùn)行,我們可以使用 os 包的 Exit 函數(shù):

習(xí)慣用法

if err != nil {
    fmt.Printf("Program stopping with error %v", err)
    os.Exit(1)
}

(此處的退出代碼 1 可以使用外部腳本獲取到)

有時(shí)候,你會(huì)發(fā)現(xiàn)這種習(xí)慣用法被連續(xù)重復(fù)地使用在某段代碼中。

當(dāng)沒有錯(cuò)誤發(fā)生時(shí),代碼繼續(xù)運(yùn)行就是唯一要做的事情,所以 if 語(yǔ)句塊后面不需要使用 else 分支。

示例 2:我們嘗試通過(guò) os.Open 方法打開一個(gè)名為 name 的只讀文件:

f, err := os.Open(name)
if err != nil {
    return err
}
doSomething(f) // 當(dāng)沒有錯(cuò)誤發(fā)生時(shí),文件對(duì)象被傳入到某個(gè)函數(shù)中
doSomething

練習(xí) 5.1 嘗試改寫 string_conversion2.go 中的代碼,要求使用 := 方法來(lái)對(duì) err 進(jìn)行賦值,哪些地方可以被修改?

示例 3:可以將錯(cuò)誤的獲取放置在 if 語(yǔ)句的初始化部分:

習(xí)慣用法

if err := file.Chmod(0664); err != nil {
    fmt.Println(err)
    return err
}

示例 4:或者將 ok-pattern 的獲取放置在 if 語(yǔ)句的初始化部分,然后進(jìn)行判斷:

習(xí)慣用法

if value, ok := readData(); ok {
…
}

注意事項(xiàng)

如果您像下面一樣,沒有為多返回值的函數(shù)準(zhǔn)備足夠的變量來(lái)存放結(jié)果:

func mySqrt(f float64) (v float64, ok bool) {
    if f < 0 { return } // error case
    return math.Sqrt(f),true
}

func main() {
    t := mySqrt(25.0)
    fmt.Println(t)
}

您會(huì)得到一個(gè)編譯錯(cuò)誤:multiple-value mySqrt() in single-value context。

正確的做法是:

t, ok := mySqrt(25.0)
if ok { fmt.Println(t) }

注意事項(xiàng) 2

當(dāng)您將字符串轉(zhuǎn)換為整數(shù)時(shí),且確定轉(zhuǎn)換一定能夠成功時(shí),可以將 Atoi 函數(shù)進(jìn)行一層忽略錯(cuò)誤的封裝:

func atoi (s string) (n int) {
    n, _ = strconv.Atoi(s)
    return
}

實(shí)際上,fmt 包(第 4.4.3 節(jié))最簡(jiǎn)單的打印函數(shù)也有 2 個(gè)返回值:

count, err := fmt.Println(x) // number of bytes printed, nil or 0, error

當(dāng)打印到控制臺(tái)時(shí),可以將該函數(shù)返回的錯(cuò)誤忽略;但當(dāng)輸出到文件流、網(wǎng)絡(luò)流等具有不確定因素的輸出對(duì)象時(shí),應(yīng)該始終檢查是否有錯(cuò)誤發(fā)生(另見練習(xí) 6.1b)。

鏈接