網(wǎng)站首頁 編程語言 正文
1. 使用ioutil讀取文本
// 全部讀取后按換行拆分 func ReadFile1(path string) error { fileHanle,err := os.OpenFile(path, os.O_RDONLY, 0666) if err != nil { return err } defer fileHanle.Close() readBytes, err := ioutil.ReadAll(fileHanle) if err != nil { return err } results := strings.Split(string(readBytes), "\n") fmt.Printf("read result:%v", results) return nil }
實現(xiàn)方式:使用iouitl一次性讀取全部文件內(nèi)容,然后使用"\n"進(jìn)行分割成行。
這種實現(xiàn)最簡單,但是只適合都內(nèi)容比較小的文件,當(dāng)讀取大文件的時候,一次讀到內(nèi)存需要占用比較大的內(nèi)存。
2.?使用bufio.Reader的ReadLine讀取
func ReadFile2(path string) error { fileHanle,err := os.OpenFile(path, os.O_RDONLY, 0666) if err != nil { return err } defer fileHanle.Close() reader := bufio.NewReader(fileHanle) var results []string // 按行處理txt for { line, _, err := reader.ReadLine() if err == io.EOF { break } results = append(results, string(line)) } fmt.Printf("read result:%v\n", results) return nil }
實現(xiàn)方式:使用NewReader創(chuàng)建bufio.Reader,循環(huán)調(diào)用Reader的ReadLine按行讀取,直接讀到文件結(jié)束標(biāo)記EOF。
bufio.Reader封裝了io, 并實現(xiàn)了緩沖I/O,同時它也實現(xiàn)了io.Reader的方法的Read方法。bufio緩沖區(qū)有默認(rèn)大小是4K。
從ReadLine返回的文本不包括行尾(“\r\n”或“\n”)。
如果一行大于緩存,isPrefix 會被設(shè)置為 true,同時返回該行的開始部分(等于緩存大小的部分)。該行剩余的部分就會在下次調(diào)用的時候返回。當(dāng)下次調(diào)用返回該行剩余部分時,isPrefix 將會是 false 。
bufio.Reader的ReadLine最終調(diào)用的是ReadSlice方法,而ReadSlice返回的[]byte是指向Reader 中的buffer的一個slice,而不是copy一份返回,所以讀取的slice可能會被一下讀取操作重新,所以官方建議是使用ReadBytes和ReadString方法。
要注意是ReadBytes和ReadString返回的結(jié)果中包含傳入的界定符,如果最終結(jié)果不需要界定符的話需要自己處理。
bufio.Reader除了有ReadLine按行讀取外,他還封裝了按指定標(biāo)記分割的方法。如下圖
3.使用bufio.Scanner讀取
func ReadFile3(path string) error { fileHanle,err := os.OpenFile(path, os.O_RDONLY, 0666) if err != nil { return err } defer fileHanle.Close() scanner := bufio.NewScanner(fileHanle) var results []string // 按行處理txt for scanner.Scan(){ lineTxt := strings.TrimSpace(scanner.Text()) if len(lineTxt) == 0 { continue } results = append(results, lineTxt) } fmt.Printf("read result:%v\n", results) return nil }
實現(xiàn)方式:使用NewScanner創(chuàng)建bufio.Scanner,使用循環(huán)調(diào)用scanner的Scan判斷是否掃描到數(shù)據(jù),然后通過scannner.Text()方法獲取到掃描的字符串。
bufio.Scanner它底層封裝了io.Reader, 它的實現(xiàn)就跟Scanner名稱一樣,是一個按字節(jié)流掃描的掃描器,當(dāng)掃描到滿足Split函數(shù)條件的字節(jié)數(shù)據(jù)后,就直接返回對應(yīng)的掃描到的內(nèi)容。
默認(rèn)情況下,它64k行限制,如果想更大,可以自己通過Buffer函數(shù)進(jìn)行設(shè)置。
Scanner默認(rèn)提供了以下方法:
Scanner
?類型具有?Split
?函數(shù),該函數(shù)接受?SplitFunc
?函數(shù)來確定?Scanner
?如何拆分給定的字節(jié)片。默認(rèn)的?SplitFunc
?是?ScanLines
,它將返回文本的每一行,并刪除行尾標(biāo)記。Split的函數(shù)定義如下:
type SplitFunc func(data []byte, atEOF bool) (advance int, token []byte, err error)
我們可以自定義實現(xiàn)SpiteFunc來實現(xiàn)不同的拆分方式,比如我們可以使用bufio.ScanWords實現(xiàn)方式來按單詞拆分,如下:
func WordCounter(){ const input = "Now is the winter of our discontent,\nMade glorious summer by this sun of York.\n" scanner := bufio.NewScanner(strings.NewReader(input)) scanner.Split(bufio.ScanWords) count := 0 for scanner.Scan() { count++ } if err := scanner.Err(); err != nil { fmt.Fprintln(os.Stderr, "reading input:", err) } fmt.Printf("%d\n", count) }
我可以跟蹤到bufio包scan.go的文件可以看到ScanWords的實現(xiàn)代碼如下:
func ScanWords(data []byte, atEOF bool) (advance int, token []byte, err error) { // Skip leading spaces. start := 0 for width := 0; start < len(data); start += width { var r rune r, width = utf8.DecodeRune(data[start:]) if !isSpace(r) { break } } // Scan until space, marking end of word. for width, i := 0, start; i < len(data); i += width { var r rune r, width = utf8.DecodeRune(data[i:]) if isSpace(r) { return i + width, data[start:i], nil } } // If we're at EOF, we have a final, non-empty, non-terminated word. Return it. if atEOF && len(data) > start { return len(data), data[start:], nil } // Request more data. return start, nil, nil }
該函數(shù)簽名和SplitFunc定義實現(xiàn)一致。
總結(jié)
原文鏈接:https://blog.csdn.net/keenw/article/details/125149112
相關(guān)推薦
- 2022-08-13 服務(wù)器上Redis主從復(fù)制和哨兵機制的配置
- 2022-08-25 .net?core中高效的動態(tài)內(nèi)存管理方案_實用技巧
- 2022-07-26 二分搜索防止整形溢出
- 2022-04-09 Android項目中g(shù)radle的執(zhí)行流程_Android
- 2022-10-15 Qt實現(xiàn)簡單UDP通信_C 語言
- 2022-05-09 Python?Matplotlib條形圖之垂直條形圖和水平條形圖詳解_python
- 2023-07-05 Redis性能瓶頸:如何優(yōu)化大key問題?
- 2022-07-12 tk mybatis報錯:org.apache.ibatis.binding.BindingExce
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支