本文主要介绍了Go语言中文件读写的相关操作。
文件是什么?
计算机中的文件是存储在外部介质(通常是磁盘)上的数据集合,文件分为文本文件和二进制文件。

1. 打开和关闭文件

os.Open()函数能够只读打开一个文件,返回一个*File和一个err。对得到的文件实例调用close()方法能够关闭文件。

  1. package main
  2. import (
  3. "fmt"
  4. "os"
  5. )
  6. func main() {
  7. // 只读方式打开当前目录下的main.go文件
  8. file, err := os.Open("./main.go")
  9. if err != nil {
  10. fmt.Println("open file failed!, err:", err)
  11. return
  12. }
  13. fmt.Println(file)
  14. // 关闭文件
  15. file.Close()
  16. }

为了防止文件忘记关闭,我们通常使用defer注册文件关闭语句。
file.Close() 一定要在错误判断之后执行,因为如果出错:file是空指针,会触发panic。
— 也就是说对有错误返回值的情况下面紧接错误判断,再执行其他代码,以免panic。
image.png
#此处go内部在Close对file做了一层判断,优化之后不会报错。但使用其他第三方库,需要注意。

  1. func (file *File) Close() error {
  2. if file == nil {
  3. return ErrInvalid
  4. }
  5. return file.file.close()
  6. }

2. 读取文件

2.1 file.Read()

基本使用

Read方法定义如下:

  1. func (f *File) Read(b []byte) (n int, err error)

它接收一个字节切片,返回读取的字节数和可能的具体错误,读到文件末尾时会返回0io.EOF。 举个例子:

  1. func main() {
  2. // 只读方式打开当前目录下的main.go文件
  3. file, err := os.Open("./main.go")
  4. if err != nil {
  5. fmt.Println("open file failed!, err:", err)
  6. return
  7. }
  8. defer file.Close()// 注册文件关闭
  9. // 使用Read方法读取数据:每次读取128个字节
  10. var tmp = make([]byte, 128)
  11. n, err := file.Read(tmp)
  12. if err == io.EOF {
  13. fmt.Println("文件读完了")
  14. return
  15. }
  16. if err != nil {
  17. fmt.Println("read file failed, err:", err)
  18. return
  19. }
  20. fmt.Printf("读取了%d字节数据\n", n)
  21. fmt.Println(string(tmp[:n]))// 通过打印整个切片看出读取的内容
  22. }

循环读取

使用for循环读取文件中的所有数据。 — 当读完或者出错跳出for循环

  1. func main() {
  2. // 只读方式打开当前目录下的main.go文件
  3. file, err := os.Open("./main.go")
  4. if err != nil {
  5. fmt.Println("open file failed!, err:", err)
  6. return
  7. }
  8. defer file.Close()
  9. // 循环读取文件
  10. var content []byte
  11. var tmp = make([]byte, 128)
  12. for {
  13. n, err := file.Read(tmp)
  14. if err == io.EOF {
  15. fmt.Println("文件读完了")
  16. break
  17. }
  18. if err != nil {
  19. fmt.Println("read file failed, err:", err)
  20. return
  21. }
  22. content = append(content, tmp[:n]...)
  23. }
  24. fmt.Println(string(content))
  25. }

2.2 bufio读取文件 - 一行一行读取

bufio是在file的基础上封装了一层API,支持更多的功能。
bufio.NewReader返回一个reader来通过reader.ReadString(‘\n’)一行一行读取。

  1. package main
  2. import (
  3. "bufio"
  4. "fmt"
  5. "io"
  6. "os"
  7. )
  8. // bufio按行读取示例
  9. func main() {
  10. file, err := os.Open("./xx.txt")
  11. if err != nil {
  12. fmt.Println("open file failed, err:", err)
  13. return
  14. }
  15. defer file.Close()
  16. reader := bufio.NewReader(file)
  17. for {
  18. line, err := reader.ReadString('\n') //注意是字符
  19. if err == io.EOF {
  20. if len(line) != 0 {
  21. fmt.Println(line)
  22. }
  23. fmt.Println("文件读完了")
  24. break
  25. }
  26. if err != nil {
  27. fmt.Println("read file failed, err:", err)
  28. return
  29. }
  30. fmt.Print(line)
  31. }
  32. }

2.3 ioutil读取整个/完整文件

io/ioutil包的ReadFile方法能够读取完整的文件,只需要将文件名作为参数传入。

  1. package main
  2. import (
  3. "fmt"
  4. "io/ioutil"
  5. )
  6. // ioutil.ReadFile读取整个文件
  7. func main() {
  8. content, err := ioutil.ReadFile("./main.go")
  9. if err != nil {
  10. fmt.Println("read file failed, err:", err)
  11. return
  12. }
  13. fmt.Println(string(content))
  14. }

3. 文件写入操作

os.OpenFile()函数能够以指定模式打开文件,从而实现文件写入相关功能。

  1. func OpenFile(name string, flag int, perm FileMode) (*File, error) {
  2. ...
  3. }

其中:

  • name:要打开的文件名
  • flag:打开文件的模式。 模式有以下几种: | 模式 | 含义 | | :—-: | :—-: | | os.O_WRONLY | 只写 | | os.O_CREATE | 创建文件 | | os.O_RDONLY | 只读 | | os.O_RDWR | 读写 | | os.O_TRUNC | 清空 | | os.O_APPEND | 追加 |

  • perm:文件权限,一个八进制数。r(读)04,w(写)02,x(执行)01。

    3.1 Write和WriteString-直接对*File写入

    1. func main() {
    2. file, err := os.OpenFile("xx.txt", os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0666)
    3. if err != nil {
    4. fmt.Println("open file failed, err:", err)
    5. return
    6. }
    7. defer file.Close()
    8. str := "hello 沙河"
    9. file.Write([]byte(str)) //写入字节切片数据
    10. file.WriteString("hello 小王子") //直接写入字符串数据
    11. }

    3.2 bufio.NewWriter-对*File处理后得到一个writer写入

    先储存在缓冲区,等达到一定阈值通过Flush写入磁盘。
    优点是:提升性能
    缺点是:一旦未Flush,白做了

    1. func main() {
    2. file, err := os.OpenFile("xx.txt", os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0666)
    3. if err != nil {
    4. fmt.Println("open file failed, err:", err)
    5. return
    6. }
    7. defer file.Close()
    8. writer := bufio.NewWriter(file)
    9. for i := 0; i < 10; i++ {
    10. writer.WriteString("hello沙河\n") //将数据先写入缓存
    11. }
    12. writer.Flush() //将缓存中的内容写入文件
    13. }

    3.3 ioutil.WriteFile-直接通过文件路径写入,没有用到os.Openfile事先打开

    1. func main() {
    2. str := "hello 沙河"
    3. err := ioutil.WriteFile("./xx.txt", []byte(str), 0666)
    4. if err != nil {
    5. fmt.Println("write file failed, err:", err)
    6. return
    7. }
    8. }

    4. 练习

    copyFile

    借助io.Copy()实现一个拷贝文件函数。

  • io.Copy() 的返回值:拷贝的字节数和error

    1. // CopyFile 拷贝文件函数
    2. func CopyFile(dstName, srcName string) (written int64, err error) {
    3. // ** 以读方式打开源文件
    4. src, err := os.Open(srcName)
    5. if err != nil {
    6. fmt.Printf("open %s failed, err:%v.\n", srcName, err)
    7. return
    8. }
    9. defer src.Close()
    10. // ** 以写|创建的方式打开目标文件
    11. dst, err := os.OpenFile(dstName, os.O_WRONLY|os.O_CREATE, 0644)
    12. if err != nil {
    13. fmt.Printf("open %s failed, err:%v.\n", dstName, err)
    14. return
    15. }
    16. defer dst.Close()
    17. return io.Copy(dst, src) //调用io.Copy()拷贝内容,返回值为拷贝的字节数和error
    18. }
    19. func main() {
    20. _, err := CopyFile("dst.txt", "src.txt")
    21. if err != nil {
    22. fmt.Println("copy file failed, err:", err)
    23. return
    24. }
    25. fmt.Println("copy done!")
    26. }

    文件中间插入内容

  1. // func (*File) Seek
  2. func (f *File) Seek(offset int64, whence int) (ret int64, err error)
  3. /* Seek设置下一次读/写的位置。offset为相对偏移量,
  4. 而whence决定相对位置:0为相对文件开头,1为相对当前位置,2为相对文件结尾。
  5. 它返回新的偏移量(相对开头)和可能的错误。*/

临时文件方案:

  1. package main
  2. import (
  3. "fmt"
  4. "io"
  5. "os"
  6. )
  7. // 指定位置插入
  8. func writeLocation() {
  9. // 打开源文件
  10. fileObj, err := os.OpenFile("./src.txt", os.O_RDWR, 0644)
  11. if err != nil {
  12. fmt.Printf("open failed, err:%v.\n", err)
  13. return
  14. }
  15. // defer fileObj.Close()
  16. // 因为没办法直接在原文件中间插入内容,因此借助临时文件实现
  17. tempFile, err := os.OpenFile("./temp.txt", os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0644)
  18. if err != nil {
  19. fmt.Printf("open failed, err:%v.\n", err)
  20. return
  21. }
  22. // defer tempFile.Close()
  23. // 读取插入位置光标前的内容写入临时文件
  24. var startOfFile [1]byte
  25. n, err := fileObj.Read(startOfFile[:])
  26. if err != nil {
  27. fmt.Printf("read from file's start failed, err:%v.\n", err)
  28. return
  29. }
  30. tempFile.Write(startOfFile[:n])
  31. fmt.Println("开始", string(startOfFile[:n]))
  32. // 再写入待插入内容
  33. var middleOfFile = []byte{'b'}
  34. fmt.Println("中间待插入", string(middleOfFile))
  35. tempFile.Write(middleOfFile)
  36. // 原文件后续内容写入临时文件
  37. var endOfFile [1024]byte
  38. // for n,err:=fileObj.Read(end_file[:]);err!=nil;{}
  39. for {
  40. n, err := fileObj.Read(endOfFile[:])
  41. if err == io.EOF {
  42. tempFile.Write(endOfFile[:n])
  43. fmt.Println("结尾内容", string(endOfFile[:n]))
  44. break
  45. }
  46. if err != nil {
  47. fmt.Printf("read from file's end failed, err:%v.\n", err)
  48. return
  49. }
  50. tempFile.Write(endOfFile[:n])
  51. fmt.Println("结尾内容", string(endOfFile[:n]))
  52. }
  53. // 关闭文件,并重命名覆盖
  54. fileObj.Close()
  55. tempFile.Close()
  56. // 临时文件重命名为原文件名字->实现覆盖
  57. os.Rename("./temp.txt", "./src.txt")
  58. }
  59. func main() {
  60. writeLocation()
  61. }

实现一个cat命令

使用文件操作相关知识,模拟实现linux平台cat命令的功能。— 连接文件并打印到标准输出设备上

  1. package main
  2. import (
  3. "bufio"
  4. "flag"
  5. "fmt"
  6. "io"
  7. "os"
  8. )
  9. // cat命令实现
  10. func cat(r *bufio.Reader) {
  11. for {
  12. buf, err := r.ReadBytes('\n') //注意是字符
  13. if err == io.EOF {
  14. break
  15. }
  16. fmt.Fprintf(os.Stdout, "%s", buf)
  17. }
  18. }
  19. func main() {
  20. flag.Parse() // 解析命令行参数
  21. if flag.NArg() == 0 {
  22. // 如果没有参数默认从标准输入读取内容
  23. cat(bufio.NewReader(os.Stdin))
  24. }
  25. // 依次读取每个指定文件的内容并打印到终端
  26. for i := 0; i < flag.NArg(); i++ {
  27. f, err := os.Open(flag.Arg(i))
  28. if err != nil {
  29. fmt.Fprintf(os.Stdout, "reading from %s failed, err:%v\n", flag.Arg(i), err)
  30. continue
  31. }
  32. cat(bufio.NewReader(f))
  33. }
  34. }