Golang 中的 Strings 包詳解之 Strings.Reader
strings.Reader
strings.Reader 是一個(gè)實(shí)現(xiàn)了 io.Reader、io.Writer、io.ByteReader、io.ByteScanner、io.RuneReader 、io.RuneScanner,、io.Seeker 和 io.WriterTo 接口的結(jié)構(gòu)體,用于從字符串中高效讀取數(shù)據(jù)。strings.Reader 可以將一個(gè)字符串包裝成一個(gè)可讀流,方便地將字符串中的數(shù)據(jù)讀取到應(yīng)用程序中。結(jié)構(gòu)體定義和對(duì)應(yīng)的方法如下:
type Reader struct {
s string
i int64 // current reading index
prevRune int // index of previous rune; or < 0
}
func (r *Reader) Len() int
func (r *Reader) Size() int64
func (r *Reader) Read(b []byte) (n int, err error)
func (r *Reader) ReadAt(b []byte, off int64) (n int, err error)
func (r *Reader) ReadByte() (byte, error)
func (r *Reader) UnreadByte() error
func (r *Reader) ReadRune() (ch rune, size int, err error)
func (r *Reader) UnreadRune() error
func (r *Reader) Seek(offset int64, whence int)
func (r *Reader) WriteTo(w io.Writer) (n int64, err error)
func (r *Reader) Reset(s string)
func NewReader(s string) *Reader
其中比較常用的方法有:
- NewReader() :返回一個(gè)從字符串 s 中讀取數(shù)據(jù)的 *Reader。
- Read(b []byte) (n int, err error): 從 strings.Reader 中讀取 len(b) 個(gè)字節(jié), 存入 b 中,返回實(shí)際讀取的字節(jié)數(shù)和可能發(fā)生的錯(cuò)誤。
- func (r *Reader) Len() int: 返回還有多少字節(jié)可以被讀取。
優(yōu)勢(shì)
可以看到,strings.Reader 包含了一個(gè)字符串、當(dāng)前讀取的位置索引和一個(gè)內(nèi)部實(shí)現(xiàn)的可讀流。strings.Reader 類型可以很方便高效地讀取一個(gè)字符串中的內(nèi)容,在讀取的過程中,Reader 會(huì)保存已讀取位置索引,位置索引就是下一次讀取的起始位置索引。Reader正是依靠這樣一個(gè)位置索引以及針對(duì)字符串值的切片表達(dá)式來實(shí)現(xiàn)快速讀取的。
使用示例
簡(jiǎn)單使用示例如下:
package main
import (
"fmt"
"strings"
)
func main() {
// 創(chuàng)建一個(gè) strings.Reader 實(shí)例
r := strings.NewReader("路多辛的所思所想")
// 讀取數(shù)據(jù)
b := make([]byte, 9)
n, err := r.Read(b)
if err != nil {
fmt.Println("讀取數(shù)據(jù)失?。?, err)
return
}
fmt.Printf("讀取了 %d 個(gè)字節(jié):%s\n", n, string(b[:n]))
}
運(yùn)行示例代碼,輸出如下:
$ go run main.go
讀取了 9 個(gè)字節(jié):路多辛
通過 strings.NewReader 創(chuàng)建了一個(gè) strings.Reader 實(shí)例后,可以使用 Read 方法從字符串中讀取數(shù)據(jù)。由于 strings.Reader 實(shí)現(xiàn)了 io.Reader 接口,因此可以使用各種基于 io.Reader 的流式處理方法,比如 io.LimitReader、io.PipeReader 等。
小結(jié)
strings.Reader 實(shí)現(xiàn)了 io.Reader、io.Writer、io.ByteReader、io.ByteScanner、io.RuneReader 、io.RuneScanner,、io.Seeker 和 io.WriterTo 接口,所以可以方便地讀取和處理字符串?dāng)?shù)據(jù)。
strings.Reader 讀取字符串也很高效,主要體現(xiàn)在對(duì)字符串的讀取機(jī)制上。在讀取的過程中,會(huì)保存當(dāng)前讀取的位置索引,位置索引就是下一次讀取的起始位置索引。