一文掌握 Go 文件的读取操作

举报
陈明勇0423 发表于 2023/04/30 19:19:41 2023/04/30
【摘要】 本文先是对 `os` 包 和 `bufio` 包进行简单的介绍,然后介绍 `os` 包里的 `Open` 和 `OpenFile` 函数以及 `File` 结构体的方法 `Read`,通过案例展示它们的使用方式。紧接着引出 `bufio` 包里的 `NewReader` 函数和 `Reader` 结构体的方法 `ReadString`,读取文件的操作推荐使用它们,能减少对磁盘的操作,高效读取数据。

作者:陈明勇
专注分享后端知识,如果文章对您有帮助,欢迎点赞收藏加关注,一起学习,一起进步!

os 包 和 bufio 包

Go 标准库的 os 包,为我们提供很多操作文件的函数,如 Open(name) 打开文件、Create(name) 创建文件等函数,与之对应的是 bufio 包,os 包是直接对磁盘进行操作的,而 bufio 包则是带有缓冲的操作,不用每次都去操作磁盘。

os.Open 与 os.OpenFile 以及 File.Read

  • Open(name string) (*File, error)

    通过 文件名文件路径+文件名 的形式打开一个文件,此文件只能用于读操作,如果文件不存在则返回 PathError

    • 参数 name文件名文件路径+文件名
    • 返回值 *File 为一个 File 结构体的指针类型,通过指针可以对文件进行读写等操作。
    • 返回值
      error 为打开文件的过程中产生的错误。
  • OpenFile(name string, flag int, perm FileMode) (*File, error)

    通过指定 文件名文件路径+文件名、文件操作模式、文件权限三个参数打开一个文件,之后可对此文件进行读写操作。

    • 参数 name文件名文件路径+文件名
    • 参数 flag 为指定文件操作模式,可选值有 O_RDONLY → 只读操作、O_WRONLY → 只写操作、O_RDWR → 读写操作、O_APPEND → 写入时向文件追加数据、O_CREATE → 如果不存在,则创建一个新文件等。
    • 参数 perm 参数表示文件的模式和权限,例如 0666 为读写权限。如果对文件权限所对应的数字不了解,可以去学习一下。
    • 返回值 *File 为一个 File 结构体的指针类型,通过指针可以对文件进行读写等操作。
    • 返回值 error 为打开文件的过程中产生的错误。
  • File.Read(b []byte) (n int, err error)

    读取与 b 等长度的字节,并存储到 b 里面。

    • 参数 b 为一个切片数组,用于指定读取长度和存储字节数据。
    • 返回值 n 为所读取字节的长度。
    • 返回值 error 为读取字节的过程中产生的错误。

读取文件操作

import (
	"fmt"
	"os"
)

func main() {
	file, err := os.Open("1.txt")
	if err != nil {
		fmt.Println(err)
		return
	}
	defer file.Close()
	data := make([]byte, 11)
	count, err := file.Read(data)
	if err != nil {
		return
	}
	fmt.Println("字节数据:", data)          // [72 101 108 108 111 32 119 111 114 108 100]
	fmt.Println("字符串数据:", string(data)) // Hello world
	fmt.Println("所获取字节的长度:", count)     // 11
}

执行结果:

字节数据: [72 101 108 108 111 32 119 111 114 108 100]
字符串数据: Hello world
所获取字节的长度: 11
  • 首先通过 Open 函数打开 1.txt 文件,用 file 变量接收,默认为可读模式;
  • 然后创建一个长度为 11 的字节切片,接着通过 file 变量的方法 Read 读取长度为 11 的字节数据。os.Open("1.txt") 等价于 os.OpenFile("1.txt", os.O_RDONLY, 0)
  • 最后打印读取到的数据,文件操作完毕之后,需要关闭文件 file.Close()

bufio.NewReader 和 Reader.ReadString

读取文件,建议使用 bufio.NewReaderReader.ReadString,减少磁盘的操作。

  • NewReader(rd io.Reader) *Reader
    获取一个有缓冲区的 Reader 指针变量,缓冲区默认大小为 4096 字节。通过变量可以对数据进行读操作。
    • 参数 rd 为一个接口,实现这个接口的数据类型变量都可以作为参数,例如上面提到的 File
    • 返回值 *ReaderReader 结构体的指针,通过指针可以读取缓冲区的数据。
  • ReadString(delim byte) (string, error)
    读取数据,直到第一次遇到分隔符 delim 为止。读取过程中发生错误会返回 EOF 错误信息。
    • 参数 delim 为分隔符,每次读取时遇到分隔符就会终止。
    • 第一个返回值为所读取的内容,内容包括分隔符。
    • 第二个返回值为读取过程中产生的错误信息。

读取文件操作

1.txt 文件的内容为:

Hello world
Hello Golang
Hello Gopher
import (
	"bufio"
	"fmt"
	"io"
	"os"
	"strings"
)

func main() {
	file, err := os.OpenFile("1.txt", os.O_RDONLY, 0)
	if err != nil {
		fmt.Println(err)
		return
	}
	defer file.Close()
	reader := bufio.NewReader(file)
	for {
		if lineData, err := reader.ReadString('\n'); err != nil {
			if err == io.EOF {
				// 因为是以换行符为分隔符,如果最后一行没有换行符,那么返回 io.EOF 错误时,也是可能读到数据的,因此判断一下是否读到了数据
				if len(lineData) > 0 {
					fmt.Println(lineData)
				}
				break
			}
		} else {
			fmt.Println(strings.TrimRight(lineData, "\n"))
		}
	}
}

执行结果:

Hello World
Hello Golang
Hello Gopher
  • 首先通过 OpenFile 函数打开 1.txt 文件,用 file 变量接收,指定为可读模式;
  • 然后通过 NewReader 函数创建一个缓冲区,将默认长度的字节读取到缓冲区中;
  • 接着通过 Reader 结构体的方法 ReadString,以 \n 为分隔符,按行读取数据,然后打印数据。
  • 其中有一个注意点就是,因为是以换行符为分隔符,如果最后一行没有换行符,那么返回 io.EOF 错误时,也是可能读到数据的,因此需要判断一下是否读到了数据。

小结

本文先是对 os 包 和 bufio 包进行简单的介绍,然后介绍 os 包里的 OpenOpenFile 函数以及 File 结构体的方法 Read,通过案例展示它们的使用方式。紧接着引出 bufio 包里的 NewReader 函数和 Reader 结构体的方法 ReadString,读取文件的操作推荐使用它们,能减少对磁盘的操作,高效读取数据。

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。