偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

超全總結(jié):Go 讀文件的 10 種方法

開發(fā) 后端
Go 中對(duì)文件內(nèi)容讀寫的方法,非常地多,其中大多數(shù)是基于 syscall 或者 os 庫的高級(jí)封裝,不同的庫,適用的場景又不太一樣,為免新手在這塊上裁跟頭,我花了點(diǎn)時(shí)間把這些內(nèi)容梳理了下。

[[442758]]

大家好,我是明哥。

Go 中對(duì)文件內(nèi)容讀寫的方法,非常地多,其中大多數(shù)是基于 syscall 或者 os 庫的高級(jí)封裝,不同的庫,適用的場景又不太一樣,為免新手在這塊上裁跟頭,我花了點(diǎn)時(shí)間把這些內(nèi)容梳理了下。

這篇是上篇,先介紹讀取文件的 10 種方法,過兩天再介紹寫入文件的。

 1. 整個(gè)文件讀取入內(nèi)存

直接將數(shù)據(jù)直接讀取入內(nèi)存,是效率最高的一種方式,但此種方式,僅適用于小文件,對(duì)于大文件,則不適合,因?yàn)楸容^浪費(fèi)內(nèi)存。

1.1 直接指定文件名讀取

有兩種方法

第一種:使用 os.ReadFile

  1. package main 
  2.  
  3. import ( 
  4.     "fmt" 
  5.     "os" 
  6.  
  7. func main() { 
  8.     content, err := os.ReadFile("a.txt"
  9.     if err != nil { 
  10.         panic(err) 
  11.     } 
  12.     fmt.Println(string(content)) 

第二種:使用 ioutil.ReadFile

  1. package main 
  2.  
  3. import ( 
  4.     "io/ioutil" 
  5.     "fmt" 
  6.  
  7. func main() { 
  8.     content, err := ioutil.ReadFile("a.txt"
  9.     if err != nil { 
  10.         panic(err) 
  11.     } 
  12.     fmt.Println(string(content)) 

其實(shí)在 Go 1.16 開始,ioutil.ReadFile 就等價(jià)于 os.ReadFile,二者是完全一致的

  1. // ReadFile reads the file named by filename and returns the contents. 
  2. // A successful call returns err == nil, not err == EOF. Because ReadFile 
  3. // reads the whole file, it does not treat an EOF from Read as an error 
  4. // to be reported. 
  5. // 
  6. // As of Go 1.16, this function simply calls os.ReadFile. 
  7. func ReadFile(filename string) ([]byte, error) { 
  8.     return os.ReadFile(filename) 

1.2 先創(chuàng)建句柄再讀取

如果僅是讀取,可以使用高級(jí)函數(shù) os.Open

  1. package main 
  2.  
  3. import ( 
  4. "os" 
  5. "io/ioutil" 
  6. "fmt" 
  7.  
  8. func main() { 
  9.     file, err := os.Open("a.txt"
  10.     if err != nil { 
  11.         panic(err) 
  12.     } 
  13.     defer file.Close() 
  14.     content, err := ioutil.ReadAll(file) 
  15.     fmt.Println(string(content)) 

之所以說它是高級(jí)函數(shù),是因?yàn)樗侵蛔x模式的 os.OpenFile

  1. // Open opens the named file for reading. If successful, methods on 
  2. // the returned file can be used for reading; the associated file 
  3. // descriptor has mode O_RDONLY. 
  4. // If there is an error, it will be of type *PathError. 
  5. func Open(name string) (*File, error) { 
  6.     return OpenFile(name, O_RDONLY, 0) 

因此,你也可以直接使用 os.OpenFile,只是要多加兩個(gè)參數(shù)

  1. package main 
  2.  
  3. import ( 
  4.     "fmt" 
  5.     "io/ioutil" 
  6.     "os" 
  7.  
  8. func main() { 
  9.     file, err := os.OpenFile("a.txt", os.O_RDONLY, 0) 
  10.     if err != nil { 
  11.         panic(err) 
  12.     } 
  13.     defer file.Close() 
  14.     content, err := ioutil.ReadAll(file) 
  15.     fmt.Println(string(content)) 

2. 每次只讀取一行

一次性讀取所有的數(shù)據(jù),太耗費(fèi)內(nèi)存,因此可以指定每次只讀取一行數(shù)據(jù)。方法有三種:

  • bufio.ReadLine()
  • bufio.ReadBytes('\n')
  • bufio.ReadString('\n')

在 bufio 的源碼注釋中,曾說道 bufio.ReadLine() 是低級(jí)庫,不太適合普通用戶使用,更推薦用戶使用 bufio.ReadBytes 和 bufio.ReadString 去讀取單行數(shù)據(jù)。

因此,這里不再介紹 bufio.ReadLine()

2.1 使用 bufio.ReadBytes

  1. package main 
  2.  
  3. import ( 
  4.     "bufio" 
  5.     "fmt" 
  6.     "io" 
  7.     "os" 
  8.     "strings" 
  9.  
  10. func main() { 
  11.     // 創(chuàng)建句柄 
  12.     fi, err := os.Open("christmas_apple.py"
  13.     if err != nil { 
  14.         panic(err) 
  15.     } 
  16.  
  17.     // 創(chuàng)建 Reader 
  18.     r := bufio.NewReader(fi) 
  19.  
  20.     for { 
  21.         lineBytes, err := r.ReadBytes('\n'
  22.         line := strings.TrimSpace(string(lineBytes)) 
  23.         if err != nil && err != io.EOF { 
  24.             panic(err) 
  25.         } 
  26.         if err == io.EOF { 
  27.             break 
  28.         } 
  29.         fmt.Println(line) 
  30.     } 

2.2 使用 bufio.ReadString

  1. package main 
  2.  
  3. import ( 
  4.     "bufio" 
  5.     "fmt" 
  6.     "io" 
  7.     "os" 
  8.     "strings" 
  9.  
  10. func main() { 
  11.     // 創(chuàng)建句柄 
  12.     fi, err := os.Open("a.txt"
  13.     if err != nil { 
  14.         panic(err) 
  15.     } 
  16.  
  17.     // 創(chuàng)建 Reader 
  18.     r := bufio.NewReader(fi) 
  19.  
  20.     for { 
  21.         line, err := r.ReadString('\n'
  22.         line = strings.TrimSpace(line) 
  23.         if err != nil && err != io.EOF { 
  24.             panic(err) 
  25.         } 
  26.         if err == io.EOF { 
  27.             break 
  28.         } 
  29.         fmt.Println(line) 
  30.     } 

3. 每次只讀取固定字節(jié)數(shù)

每次僅讀取一行數(shù)據(jù),可以解決內(nèi)存占用過大的問題,但要注意的是,并不是所有的文件都有換行符 \n。

因此對(duì)于一些不換行的大文件來說,還得再想想其他辦法。

3.1 使用 os 庫

通用的做法是:

  • 先創(chuàng)建一個(gè)文件句柄,可以使用 os.Open 或者 os.OpenFile
  • 然后 bufio.NewReader 創(chuàng)建一個(gè) Reader
  • 然后在 for 循環(huán)里調(diào)用 Reader 的 Read 函數(shù),每次僅讀取固定字節(jié)數(shù)量的數(shù)據(jù)。
  1. package main 
  2.  
  3. import ( 
  4.     "bufio" 
  5.     "fmt" 
  6.     "io" 
  7.     "os" 
  8.  
  9. func main() { 
  10.     // 創(chuàng)建句柄 
  11.     fi, err := os.Open("a.txt"
  12.     if err != nil { 
  13.         panic(err) 
  14.     } 
  15.  
  16.     // 創(chuàng)建 Reader 
  17.     r := bufio.NewReader(fi) 
  18.  
  19.     // 每次讀取 1024 個(gè)字節(jié) 
  20.     buf := make([]byte, 1024) 
  21.     for { 
  22.         n, err := r.Read(buf) 
  23.         if err != nil && err != io.EOF { 
  24.             panic(err) 
  25.         } 
  26.  
  27.         if n == 0 { 
  28.             break 
  29.         } 
  30.         fmt.Println(string(buf[:n])) 
  31.     } 

3.2 使用 syscall 庫

os 庫本質(zhì)上也是調(diào)用 syscall 庫,但由于 syscall 過于底層,如非特殊需要,一般不會(huì)使用 syscall

本篇為了內(nèi)容的完整度,這里也使用 syscall 來舉個(gè)例子。

本例中,會(huì)每次讀取 100 字節(jié)的數(shù)據(jù),并發(fā)送到通道中,由另外一個(gè)協(xié)程進(jìn)行讀取并打印出來。

  1. package main 
  2.  
  3. import ( 
  4.     "fmt" 
  5.     "sync" 
  6.     "syscall" 
  7.  
  8. func main() { 
  9.     fd, err := syscall.Open("christmas_apple.py", syscall.O_RDONLY, 0) 
  10.     if err != nil { 
  11.         fmt.Println("Failed on open: ", err) 
  12.     } 
  13.     defer syscall.Close(fd) 
  14.  
  15.     var wg sync.WaitGroup 
  16.     wg.Add(2) 
  17.     dataChan := make(chan []byte) 
  18.     go func() { 
  19.         wg.Done() 
  20.         for { 
  21.             data := make([]byte, 100) 
  22.             n, _ := syscall.Read(fd, data) 
  23.             if n == 0 { 
  24.                 break 
  25.             } 
  26.             dataChan <- data 
  27.         } 
  28.         close(dataChan) 
  29.     }() 
  30.  
  31.     go func() { 
  32.         defer wg.Done() 
  33.         for { 
  34.             select { 
  35.             case data, ok := <-dataChan: 
  36.                 if !ok { 
  37.                     return 
  38.                 } 
  39.  
  40.                 fmt.Printf(string(data)) 
  41.             default
  42.  
  43.             } 
  44.         } 
  45.     }() 
  46.     wg.Wait() 
本文轉(zhuǎn)載自微信公眾號(hào)「Go編程時(shí)光」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系Go編程時(shí)光公眾號(hào)。

 

責(zé)任編輯:武曉燕 來源: Go編程時(shí)光
相關(guān)推薦

2020-09-15 15:50:33

Python編程語言代碼

2024-03-01 20:55:40

Pytorch張量Tensor

2025-01-20 08:00:00

圖像增強(qiáng)深度學(xué)習(xí)AI

2024-06-11 07:34:58

C#字符串性能

2024-05-10 12:59:58

PyTorch人工智能

2020-05-15 08:28:18

二進(jìn)制文件binLinux

2022-05-31 16:00:46

Go 編程語言復(fù)制文件Go 標(biāo)準(zhǔn)庫

2017-06-09 13:51:42

Linux命令刪除文件

2019-10-24 08:52:23

Linux壓縮文件歸檔

2020-11-10 10:23:25

IT成本CIO高級(jí)分析師

2024-03-22 15:32:21

機(jī)器學(xué)習(xí)算法

2009-12-25 10:28:10

WPF路由事件

2022-01-16 09:30:34

Ansible自動(dòng)化工具開源

2020-05-06 09:51:37

二進(jìn)制Linux命令行工具

2011-09-02 14:59:15

2021-07-26 09:47:38

Go語言C++

2014-06-23 16:56:46

2015-05-27 14:55:45

2010-03-15 15:45:50

2009-10-21 17:02:51

綜合布線施工經(jīng)驗(yàn)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)