Go語(yǔ)言之io.ReadAtLeast函數(shù)的基本使用和原理解析
1. 引言
io.ReadAtLeast
函數(shù)是Go標(biāo)準(zhǔn)庫(kù)提供的一個(gè)非常好用的函數(shù),能夠指定從數(shù)據(jù)源最少讀取到的字節(jié)數(shù)。本文我們將從io.ReadAtLeast
函數(shù)的基本定義出發(fā),講述其基本使用和實(shí)現(xiàn)原理,以及一些注意事項(xiàng),基于此完成對(duì)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ù)會(huì)嘗試從讀取器中最少讀取 min
個(gè)字節(jié)的數(shù)據(jù),并將其存儲(chǔ)在 buf
中。
2.2 使用示例
下面是一個(gè)示例代碼,演示如何使用 io.ReadAtLeast
函數(shù)從標(biāo)準(zhǔn)輸入讀取至少 5 個(gè)字節(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ā)生錯(cuò)誤:", err) return } fmt.Printf("成功讀取了 %d 個(gè)字節(jié):%s\n", n, buffer) }
在這個(gè)例子中,我們創(chuàng)建了一個(gè)長(zhǎng)度為 10 的字節(jié)切片 buffer
,并使用 io.ReadAtLeast
函數(shù)從標(biāo)準(zhǔn)輸入讀取至少 5 個(gè)字節(jié)的數(shù)據(jù)到 buffer
中。下面是一個(gè)可能的輸出,具體如下:
hello,world 成功讀取了 10 個(gè)字節(jié):hello,worl
這里其指定 min
為5,也就是最少讀取5個(gè)字節(jié)的數(shù)據(jù),此時(shí)調(diào)用io.ReadAtLeast
函數(shù)一次性讀取到了10個(gè)字節(jié)的數(shù)據(jù),此時(shí)也滿足要求。這里也間接說明了io.ReadAtLeast
只保證最少要讀取min
個(gè)字節(jié)的數(shù)據(jù),但是并不限制更多數(shù)據(jù)的讀取。
3. 實(shí)現(xiàn)原理
在了解了io.ReadAtLeast
函數(shù)的基本定義和使用后,這里我們來對(duì)io.ReadAtLeast
函數(shù)的實(shí)現(xiàn)來進(jìn)行基本的說明,加深對(duì)io.ReadAtLeast
函數(shù)的理解。
其實(shí) io.ReadAtLeast
的實(shí)現(xiàn)非常簡(jiǎn)單,其定義一個(gè)變量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長(zhǎng)度 小于 最小讀取字節(jié)數(shù)min的值,此時(shí)直接返回錯(cuò)誤 if len(buf) < min { return 0, ErrShortBuffer } // 在 n < min 時(shí),不斷調(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í)數(shù)據(jù)已經(jīng)全部讀取完了,此時(shí)返回 ErrUnexpectedEOF err = ErrUnexpectedEOF } return }
4. 注意事項(xiàng)
4.1 注意無限等待情況的出現(xiàn)
從上面io.ReadAtLeast
的實(shí)現(xiàn)可以看出來,如果一直沒有讀取到指定數(shù)量的數(shù)據(jù),同時(shí)也沒有發(fā)生錯(cuò)誤,將一直等待下去,直到讀取到至少指定數(shù)量的字節(jié)數(shù)據(jù),或者遇到錯(cuò)誤為止。下面舉個(gè)代碼示例來展示下效果:
func main() { buffer := make([]byte, 5) n, err := io.ReadAtLeast(os.Stdin, buffer, 5) if err != nil { fmt.Println("讀取過程中發(fā)生錯(cuò)誤:", err) return } fmt.Printf("成功讀取了 %d 個(gè)字節(jié):%s\n", n, buffer) }
在上面代碼的例子中,會(huì)調(diào)用io.ReadAtLeast
函數(shù)從標(biāo)準(zhǔn)輸入中讀取 5 個(gè)字節(jié)的數(shù)據(jù),如果標(biāo)準(zhǔn)輸入一直沒有輸夠5個(gè)字節(jié),此時(shí)這個(gè)函數(shù)將會(huì)一直等待下去。比如下面的這個(gè)輸入,首先輸入了he
兩個(gè)字符,然后回車,由于還沒有達(dá)到5個(gè)字符,此時(shí)io.ReadAtLeast
函數(shù)一直不會(huì)返回,只有再輸入llo
這幾個(gè)字符后,才滿足5個(gè)字符,才能夠繼續(xù)執(zhí)行,所以在使用io.ReadAtLeast
函數(shù)時(shí),需要注意無限等待的情況。
he llo 成功讀取了 5 個(gè)字節(jié):he ll
4.2 確保 buf
的大小足夠容納至少 min
個(gè)字節(jié)的數(shù)據(jù)
在調(diào)用io.ReadAtLeast
函數(shù)時(shí),需要保證緩沖區(qū)buf
的大小需要滿足min
,如果緩沖區(qū)的大小比 min
參數(shù)還小的話,此時(shí)將永遠(yuǎn)滿足不了 最少讀取 min
個(gè)字節(jié)數(shù)據(jù)的要求。
從上面io.ReadAtLeast
的實(shí)現(xiàn)可以看出來,如果其發(fā)現(xiàn)buf
的長(zhǎng)度小于 min
,其也不會(huì)嘗試去讀取數(shù)據(jù),其會(huì)直接返回一個(gè)ErrShortBuffer
的錯(cuò)誤,下面通過一個(gè)代碼展示下效果:
func main() { buffer := make([]byte, 3) n, err := io.ReadAtLeast(os.Stdin, buffer, 5) if err != nil { fmt.Println("讀取過程中發(fā)生錯(cuò)誤:", err) return } fmt.Printf("成功讀取了 %d 個(gè)字節(jié):%s\n", n, buffer) }
比如上述函數(shù)中,指定的buffer
的長(zhǎng)度為3,但是io.ReadAtLeast
要求最少讀取5個(gè)字節(jié),此時(shí)buffer
并不能容納5個(gè)字節(jié)的數(shù)據(jù),此時(shí)將會(huì)直接ErrShortBuffer
錯(cuò)誤,如下:
讀取過程中發(fā)生錯(cuò)誤: short buffer
5. 總結(jié)
io.ReadAtLeast
函數(shù)是Go語(yǔ)言標(biāo)準(zhǔn)庫(kù)提供的一個(gè)工具函數(shù),能夠從數(shù)據(jù)源讀取至少指定數(shù)量的字節(jié)數(shù)據(jù)到緩沖區(qū)中。 我們先從 io.ReadAtLeast
函數(shù)的基本定義出發(fā),之后通過一個(gè)簡(jiǎn)單的示例,展示如何使用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
個(gè)字節(jié)的數(shù)據(jù)。
基于此,完成了對(duì)io.ReadAtLeast
函數(shù)的介紹,希望對(duì)你有所幫助。
到此這篇關(guān)于一文了解io.ReadAtLeast函數(shù)的文章就介紹到這了,更多相關(guān)io.ReadAtLeast函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go pprof內(nèi)存指標(biāo)含義備忘錄及案例分析
這篇文章主要介紹了Go pprof內(nèi)存指標(biāo)含義備忘錄問題,小編特此把問題及案例分享到腳本之家平臺(tái)供大家學(xué)習(xí),需要的朋友可以參考下2020-03-03go slice 擴(kuò)容實(shí)現(xiàn)原理源碼解析
這篇文章主要為大家介紹了go slice 擴(kuò)容實(shí)現(xiàn)原理源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01Golang 關(guān)于Gin框架請(qǐng)求參數(shù)的獲取方法
Gin是Go語(yǔ)言的Web框架,提供路由和中間件支持,本文介紹如何使用Gin獲取HTTP請(qǐng)求參數(shù),包括URLPath參數(shù)、URLQuery參數(shù)、HTTPBody參數(shù)和Header參數(shù),詳解直接獲取和綁定到結(jié)構(gòu)體兩種方法,幫助開發(fā)者高效處理Web請(qǐng)求2024-10-10Go語(yǔ)言實(shí)現(xiàn)超時(shí)的三種方法實(shí)例
超時(shí)在一些業(yè)務(wù)場(chǎng)景里非常普遍,下面這篇文章主要給大家介紹了關(guān)于Go語(yǔ)言實(shí)現(xiàn)超時(shí)的三種方法,文中通過實(shí)例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Go語(yǔ)言具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-07-07淺析Go中函數(shù)的健壯性,panic異常處理和defer機(jī)制
這篇文章主要為大家詳細(xì)介紹了Go中函數(shù)的健壯性,panic異常處理和defer機(jī)制的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-10-10