Golang 文件操作
Os模块的使用与源码研究
文件:计算机中的文件是存储在外部介质(通常是磁盘)上的数据集合,文件分为文本文件和二进制文件。例如咱们常见的文件后缀名.exe
,.txt
,’.word’…等等
文件的基本操作可简单分为增
、删
两类,也就是咱们所说的CURD(增删改查),也是基于此两类操作。可简单理解为打开文件夹
、CURD
、关闭文件夹。结束~
golang对于文件基本上都是基于Golang的os
模块,那让我们一起了解一下,那么Golang是如何对文件进行操作呢。Let’s Go~
打开文件
Golang中打开文件使用os.Open
模块,官方os.open部分源码如下:
1 |
// os.Open |
Open打开命名文件以供读取。如果成功,则可以使用返回文件上的方法进行读取;关联的文件。描述符的模式为O_RDONLY。 如果有错误,它将是* PathError类型。
它接收一个string 类型的变量
name
,返回两个值,File的指针和错误error。那么我们使用它打开文件的的时候就需要这样做
1
2 fileObj, err := os.Open(name string)
// 其中os.Open中的name为路径Path
基础使用的介绍暂且为止,其实我们更应该关心的应该是OpenFile(name, O_RDONLY, 0)
,这个函数到底干了啥,我们追踪一下这个函数(在GoLang编辑器中, mac可以直接使用command +
鼠标左键直接进入,Win可以使用ctrl + 鼠标左键),如下:
1 |
func OpenFile(name string, flag int, perm FileMode) (*File, error) { |
这个文件全部内容还是有点分量的,有信息的伙伴,可以详细的阅读一下全部内容。暂且为止
那让我们实践一下,使用Golang打开文件,如下
1 |
package main |
以防忘记关闭文件,造成bug,我们在这里使用defer + 关闭。
注意:在编辑器中并不建议直接使用鼠标右键运行,这样可能会导致路径错误。大部分的编辑器都并不是只运行此文件!!!
1 Open File Error Message:&os.PathError{Op:"open", Path:"./main.go", Err:0x2}如果你遇见了类似的错误,你可以直接在终端中,切换到当前路径。使用
go run main.go
,直接运行。这样就可以直接得到正确的结果啦
读取文件
打开文件之后,那么我们可以就可以对他们进行操作了,我们在这里主要演示一下读取文件的操作。还是老样子,先看一下主要的相关源码,如下:
1 |
// FileObj.Read() |
FileObj.Read()
示例化接受文件的地址值(也就是咱们前面打开获取到的结果),接受切片的字节,返回读取的内容,以及错误
在此函数中首先检查是否为有效的读取,然后在进行f.read(b)的操作,接受其返回结果。
f.read(b)
在这里,主要检测是否在读取,如果是那么返回本次的读取内容
从以上我们不难看出,其实读取文件是读取文件内部的字节
那么更具FileObj.Read(),我们可以了解它基本的使用方法,如下
1 |
func (f *File) Read(b []byte) (n int, err error) |
读取部分的示例代码如下:
在这里我们需要考虑:是否能够正常读取?是否读完了?具体请看异常处理部分
1 |
// 读取文件 |
输出结果如下:
以上很明显是并没有读完的仅读取了部分,原始的全部代码如下
1 |
package main |
完整读取
for无线循环读取
由于以上我们并没有读取完整个文件,那么我需要读取全部的该怎么办呢?一个方法是不断的读取下去,然后和在一起就是完整的内容了,示例代码如下
1 |
package main |
主要的思路为:无限循环去读取,读完了之后break掉。然后把读取的内容合并起来
这种读取虽然可行,不过是否有点太麻烦了,那么有什么更简便的方式呢?答案当然是有的,bufio读取
bufio读取
bufio是在file的基础上封装了一层API,支持更多的功能。
主要的部分源码如下所示
1 |
// bufio.NewReader |
它简便的原因是因为已经帮我们定义了文件的指针,以及它还定义了缓冲区,这样我们使用它来读取更加的快与便捷。
bufio.NewReader语法格式
1 |
func NewReader(rd io.Reader) *Reader |
使用如下
1 |
package main |
输入结果如上,略。。。
搞了这么多,就没有一键读取的么?当然也是有的,让我们来了体验一下ioutil
读取整个文件的愉悦。
1 |
package main |
其内部的实现原理,先预测整个文件的大小。然后一次性全部读取。当然需要做好异常的准备哦
1 |
// ReadFile reads the file named by filename and returns the contents. |
文件写入操作
os.OpenFile()
函数能够以指定模式打开文件,从而实现文件写入相关功能。
1 |
func OpenFile(name string, flag int, perm FileMode) (*File, error) { |
其中:
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。
Write和WriteString
1 |
func main() { |
bufio.NewWriter
1 |
func main() { |
ioutil.WriteFile
1 |
func main() { |
so cool~