一文了解io.ReadAtLeast函數(shù)

1. 引言

io.ReadAtLeast 函數(shù)是Go標(biāo)準(zhǔn)庫提供的一個非常好用的函數(shù),能夠指定從數(shù)據(jù)源最少讀取到的字節(jié)數(shù)。本文我們將從io.ReadAtLeast 函數(shù)的基本定義出發(fā),講述其基本使用和實(shí)現(xiàn)原理,以及一些注意事項(xiàng),基于此完成對io.ReadAtLeast 函數(shù)的介紹。

2. 基本說明

2.1 基本定義

io.ReadAtLeast 函數(shù)用于從讀取器(io.Reader)讀取至少指定數(shù)量的字節(jié)數(shù)據(jù)到緩沖區(qū)中。函數(shù)定義如下:

func ReadAtLeast(r Reader, buf []byte, min int) (n int, err error)

其中r 是數(shù)據(jù)源,從它讀取數(shù)據(jù),而buf是用于接收讀取到的數(shù)據(jù)的字節(jié)切片,min是要讀取的最小字節(jié)數(shù)。io.ReadAtLeast 函數(shù)會嘗試從讀取器中最少讀取 min 個字節(jié)的數(shù)據(jù),并將其存儲在 buf 中。

2.2 使用示例

下面是一個示例代碼,演示如何使用 io.ReadAtLeast 函數(shù)從標(biāo)準(zhǔn)輸入讀取至少 5 個字節(jié)的數(shù)據(jù):

package main

import (
        "fmt"
        "io"
        "os"
)

func main() {
        buffer := make([]byte, 10)

        n, err := io.ReadAtLeast(os.Stdin, buffer, 5)
        if err != nil {
                fmt.Println("讀取過程中發(fā)生錯誤:", err)
                return
        }

        fmt.Printf("成功讀取了 %d 個字節(jié):%s\n", n, buffer)
}

在這個例子中,我們創(chuàng)建了一個長度為 10 的字節(jié)切片 buffer,并使用 io.ReadAtLeast 函數(shù)從標(biāo)準(zhǔn)輸入讀取至少 5 個字節(jié)的數(shù)據(jù)到 buffer 中。下面是一個可能的輸出,具體如下:

hello,world
成功讀取了 10 個字節(jié):hello,worl

這里其指定 min 為5,也就是最少讀取5個字節(jié)的數(shù)據(jù),此時調(diào)用io.ReadAtLeast函數(shù)一次性讀取到了10個字節(jié)的數(shù)據(jù),此時也滿足要求。這里也間接說明了io.ReadAtLeast只保證最少要讀取min個字節(jié)的數(shù)據(jù),但是并不限制更多數(shù)據(jù)的讀取。

3. 實(shí)現(xiàn)原理

在了解了io.ReadAtLeast 函數(shù)的基本定義和使用后,這里我們來對io.ReadAtLeast 函數(shù)的實(shí)現(xiàn)來進(jìn)行基本的說明,加深對io.ReadAtLeast 函數(shù)的理解。

其實(shí) io.ReadAtLeast 的實(shí)現(xiàn)非常簡單,其定義一個變量n, 保存了讀取到的字節(jié)數(shù),然后不斷調(diào)用數(shù)據(jù)源Reader中的 Read 方法讀取數(shù)據(jù),然后自增變量n 的值,直到 n 大于 最小讀取字節(jié)數(shù)為止。下面來看具體代碼的實(shí)現(xiàn):

func ReadAtLeast(r Reader, buf []byte, min int) (n int, err error) {
   // 傳入的緩沖區(qū)buf長度 小于 最小讀取字節(jié)數(shù)min的值,此時直接返回錯誤
   if len(buf) < min {
      return 0, ErrShortBuffer
   }
   // 在 n < min 時,不斷調(diào)用Read方法讀取數(shù)據(jù)
   // 最多讀取 len(buf) 字節(jié)的數(shù)據(jù)
   for n < min && err == nil {
      var nn int
      nn, err = r.Read(buf[n:])
      // 自增 n 的值
      n += nn
   }
   if n >= min {
      err = nil
   } else if n > 0 && err == EOF {
      // 讀取到的數(shù)據(jù)字節(jié)數(shù) 小于 min值,同時數(shù)據(jù)已經(jīng)全部讀取完了,此時返回 ErrUnexpectedEOF
      err = ErrUnexpectedEOF
   }
   return
}

4. 注意事項(xiàng)

4.1 注意無限等待情況的出現(xiàn)

從上面io.ReadAtLeast 的實(shí)現(xiàn)可以看出來,如果一直沒有讀取到指定數(shù)量的數(shù)據(jù),同時也沒有發(fā)生錯誤,將一直等待下去,直到讀取到至少指定數(shù)量的字節(jié)數(shù)據(jù),或者遇到錯誤為止。下面舉個代碼示例來展示下效果:

func main() {
   buffer := make([]byte, 5)
   n, err := io.ReadAtLeast(os.Stdin, buffer, 5)
   if err != nil {
      fmt.Println("讀取過程中發(fā)生錯誤:", err)
      return
   }

   fmt.Printf("成功讀取了 %d 個字節(jié):%s\n", n, buffer)
}

在上面代碼的例子中,會調(diào)用io.ReadAtLeast 函數(shù)從標(biāo)準(zhǔn)輸入中讀取 5 個字節(jié)的數(shù)據(jù),如果標(biāo)準(zhǔn)輸入一直沒有輸夠5個字節(jié),此時這個函數(shù)將會一直等待下去。比如下面的這個輸入,首先輸入了he兩個字符,然后回車,由于還沒有達(dá)到5個字符,此時io.ReadAtLeast函數(shù)一直不會返回,只有再輸入llo這幾個字符后,才滿足5個字符,才能夠繼續(xù)執(zhí)行,所以在使用io.ReadAtLeast函數(shù)時,需要注意無限等待的情況。

he
llo
成功讀取了 5 個字節(jié):he
ll

4.2 確保 buf 的大小足夠容納至少 min 個字節(jié)的數(shù)據(jù)

在調(diào)用io.ReadAtLeast函數(shù)時,需要保證緩沖區(qū)buf的大小需要滿足min,如果緩沖區(qū)的大小比 min 參數(shù)還小的話,此時將永遠(yuǎn)滿足不了 最少讀取 min個字節(jié)數(shù)據(jù)的要求。

從上面io.ReadAtLeast 的實(shí)現(xiàn)可以看出來,如果其發(fā)現(xiàn)buf的長度小于 min,其也不會嘗試去讀取數(shù)據(jù),其會直接返回一個ErrShortBuffer 的錯誤,下面通過一個代碼展示下效果:

func main() {
   buffer := make([]byte, 3)
   n, err := io.ReadAtLeast(os.Stdin, buffer, 5)
   if err != nil {
      fmt.Println("讀取過程中發(fā)生錯誤:", err)
      return
   }

   fmt.Printf("成功讀取了 %d 個字節(jié):%s\n", n, buffer)
}

比如上述函數(shù)中,指定的buffer的長度為3,但是io.ReadAtLeast要求最少讀取5個字節(jié),此時buffer并不能容納5個字節(jié)的數(shù)據(jù),此時將會直接ErrShortBuffer錯誤,如下:

讀取過程中發(fā)生錯誤: short buffer

5. 總結(jié)

io.ReadAtLeast函數(shù)是Go語言標(biāo)準(zhǔn)庫提供的一個工具函數(shù),能夠從數(shù)據(jù)源讀取至少指定數(shù)量的字節(jié)數(shù)據(jù)到緩沖區(qū)中。 我們先從 io.ReadAtLeast 函數(shù)的基本定義出發(fā),之后通過一個簡單的示例,展示如何使用io.ReadAtLeast函數(shù)實(shí)現(xiàn)至少讀取指定字節(jié)數(shù)據(jù)。

接著我們講述了io.ReadAtLeast函數(shù)的實(shí)現(xiàn)原理,其實(shí)就是不斷調(diào)用源Reader的Read方法,直接讀取到的數(shù)據(jù)數(shù)滿足要求。

在注意事項(xiàng)方面,則強(qiáng)調(diào)了調(diào)用io.ReadAtLeast 可能出現(xiàn)無限等待的問題,以及需要確保 buf 的大小足夠容納至少 min 個字節(jié)的數(shù)據(jù)。

基于此,完成了對io.ReadAtLeast函數(shù)的介紹,希望對你有所幫助。

本文由博客一文多發(fā)平臺 OpenWrite 發(fā)布!

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容