Go開發工程師
未來3-5年企業高性能項目不可替代的語言,從基礎到項目實戰再到重構,真正從入門到精通
Go語言的函數可以一次返回多個結果。這就為我們溫和地報告錯誤提供了語言級別的支持。實際上,這也是Go語言中處理錯誤的慣用法之一。我們先來回顧前一小節的例子:
func readFile(path string) ([]byte, error) { file, err := os.Open(path) if err != nil { return nil, err } defer file.Close() return ioutil.ReadAll(file) }
函數readFile
有兩個結果聲明。第二個結果聲明的類型是error
。error
是Go語言內置的一個接口類型。它的聲明是這樣的:
type error interface { Error() string }
顯然,只要一個類型的方法集合包含了名為Error
、無參數聲明且僅聲明了一個string
類型的結果的方法,就相當于實現了error
接口。os.Open
函數的第二個結果值就的類型就是這樣的。我們把它賦給了變量err
。也許你已經意識到,在Go語言中,函數與其調用方之間溫和地傳遞錯誤的方法即是如此。
在調用了os.Open
函數并取得其結果之后,我們判斷err
是否為nil
。如果答案是肯定的,那么就直接把該錯誤(這里由err
代表)返回給調用方。這條if
語句實際上是一條衛述語句。這樣的語句會檢查流程中的某個步驟是否存在異常,并在必要時中止流程并報告給上層的程序(這里是調用方)。在Go語言的標準庫以及很多第三方庫中,我們經常可以看到這樣的代碼。我們也建議大家在自己的程序中善用這樣的衛述語句。
現在我們把目光聚焦到readFile
函數中的最后一條語句上。這是一條return
語句。它把對ioutil.ReadAll
函數的調用的結果直接作為readFile
函數的結果返回了。實際上,ioutil.ReadAll
函數的結果聲明列表與readFile
的結果聲明列表是一致的。也就是說,它們聲明的結果的數量、類型和順序都是相同的。因此,我們才能夠做這種返回結果上的“嫁接”。這又是一個Go語言編碼中的慣用法。
好了,在知曉怎樣在傳遞錯誤之后,讓我們來看看怎樣創造錯誤。沒錯,在很多時候,我們需要創造出錯誤(即error
類型的值)并把它傳遞給上層程序。這很簡單。只需調用標準庫代碼包errors
的New
函數即可。例如,我們只要在readFile
函數的開始處加入下面這段代碼就可以更快的在參數值無效時告知調用方:
if path == "" { return nil, errors.New("The parameter is invalid!") }
errors.New
是一個很常用的函數。在Go語言標準庫的代碼包中有很多由此函數創建出來的錯誤值,比如os.ErrPermission
、io.EOF
等變量的值。我們可以很方便地用操作符==
來判斷一個error
類型的值與這些變量的值是否相等,從而來確定錯誤的具體類別。就拿io.EOF
來說,它代表了一個信號。該信號用于通知數據讀取方已無更多數據可讀。我們在得到這樣一個錯誤的時候不應該把它看成一個真正的錯誤,而應該只去結束相應的讀取操作。請看下面的示例:
br := bufio.NewReader(file) var buf bytes.Buffer for { ba, isPrefix, err := br.ReadLine() if err != nil { if err == io.EOF { break } fmt.Printf("Error: %s\n", err) break } buf.Write(ba) if !isPrefix { buf.WriteByte('\n') } }
可以看到,這段代碼使用到了前面示例中的變量file
。它的功能是把file
代表的文件中的所有內容都讀取到一個緩沖器(由變量buf
代表)中。請注意,該示例中的第6~8行代碼。如果判定err
代表的錯誤值等于io.EOF
的值(即它們是同一個值),那么我們只需退出當前的循環以使讀取操作結束即可。
總之,只要能夠善用error
接口、errors.New
函數和比較操作符==
,我們就可以玩兒轉Go語言中的一般錯誤處理。
命令源碼文件index.go的功能是讀取自己的內容。這其中用到了readFile
函數。請注意,我把這個函數中的最后一條語句替換成了return read(file)
。你現在的任務是,結合本節“知識要點”中的最后一段代碼編寫出一個完整的read
函數,使得該文件可以正確地被運行,并在標準輸出打印出index.go文件的內容。
該文件中的導入語句中已列出了所有可能會用到的標準庫代碼包。在Go語言官方的文檔網站上可以查到其中各種程序實體的用法。
注意:由于本編程課件中無法模擬真實的程序運行環境,所以它讀取index.go文件的操作是不會成功的。請你把index.go中的代碼(包括你寫的read
函數)復制到一臺Linux或Mac機器上再加以運行。
read
函數的實現可以是:
func read(r io.Reader) ([]byte, error) { br := bufio.NewReader(r) var buf bytes.Buffer for { ba, isPrefix, err := br.ReadLine() if err != nil { if err == io.EOF { break } return nil, err } buf.Write(ba) if !isPrefix { buf.WriteByte('\n') } } return buf.Bytes(), nil }
請驗證,完成請求
由于請求次數過多,請先驗證,完成再次請求
打開微信掃碼自動綁定
綁定后可得到
使用 Ctrl+D 可將課程添加到書簽
舉報