在Golang中我们操作文件时,需要先打开文件,打开文件操作完毕后,还需要关闭文件,如果只打开文件,不关闭文件,会造成系统资源的浪费。
在Golang中打开文件使用Open函数,关闭文件使用Close函数,打开文件、关闭文件以及大多数文件操作都涉及一个很重要的结构体os.File结构体。
1.1 os.File结构体
type File struct {
*file // os specific
}
type file struct {
pfd poll.FD
name string
dirinfo *dirInfo // nil unless directory being read
appendMode bool // whether file is opened for appending
}
说明:
这里可以看到os.File结构体里面包含了一个file指针,file指针结构体有四个成员,分别为:
- pfd:是一个FD结构体类型,是一个文件的唯一标志,每一个被打开的文件在操作系统中,都会有一个文件标志符,来唯一标识一个文件,就是这里的pfd.
- name:文件名
- dirinfo:文件的路径信息,也是一个结构体
- appendMode:是一个bool类型,表明该文件是否可以被追加写入内容
1.2 Open函数
语法:
func Open(name string) (*File, error)
参数:
- name:文件名
返回值:
- File:打开的文件句柄
- err:如果打开失败,则返回错误信息,否则返回nil
说明
Open函数接受一个字符串类型的文件名作为参数,如果打开成功,则返回一个File结构体的指针,否则就返回error错误信息。
1.3 Close函数
语法:
func (file *File) Close() error
参数:
- file:打开的文件
返回值 - error:如果打开失败则返回错误信息,否则返回nil
说明:
使用File指针来调用Close函数,如果关闭失败,则返回error错误信息。
1.4 示例说明
使用Open函数打开文件,使用Close函数关闭文件:
package main
import (
"fmt"
"os"
)
func main() {
fmt.Println("Open File Test")
fileName := "D:/go项目/test.go"
file, err := os.Open(fileName)
if err != nil {
fmt.Println("Open file err:", err)
return
}
fmt.Println("Open File Sucess")
if err := file.Close(); err != nil {
fmt.Println("Close File Err:", err)
return
}
fmt.Println("Close File Success")
}
二.Golang读取文件
Golang中,读取文件有四种方法:分别为:
- ioutil.ReadFile读取文件
- file.Read读取文件
- bufio.NewReader读取文件
- ioutil.ReadAll读取文件
2.1 ioutil.ReadFile读取文件
语法:
func ReadFile(filename string) ([]byte, error)
参数:
-
filename:文件名
返回值: -
[]byte:读取到的文件内容
-
error:如果读取失败,返回错误信息,否则返回nil
说明:
ioutil.ReadFile读取文件只需要传入一个文件名作为参数,读取成功,会将文件的内容作为一个字节数组返回,如果读取错误,返回error信息。
使用ReadFile读取文件不需要手动打开与关闭文件,打开与关闭文件的动作,系统自动帮我们完成,同时使用ReadFile读取文件时,只适合读取小文件,不适合读取大文件。 -
示例说明
package main
import (
"fmt"
"io/ioutil"
)
func main() {
fmt.Println("ioutil.ReadFile test")
fileName := "D:/go项目/test.go"
fileData, err := ioutil.ReadFile(fileName)
if err != nil {
fmt.Println("read file err:", err)
}
fmt.Println(string(fileData))
}
2.2 file.Read读取文件
语法:
func (f *File) Read(b []byte) (n int, err error)
参数:
- f:打开的文件句柄
- b:读取文件内容存放的指针
返回值: - n:读取到的字节数
- err:如果读取失败,返回错误信息,否则返回nil
说明:
使用file.Read读取文件时,首先我们需要打开文件,接着使用打开的文件返回的文件句柄来读取文件,文件读取结束的标志时返回的n等于0,如果我们需要读取整个文件内容,那么我们需要使用for循环不停的读取文件,直到n等于0.
2.3 NewReader读取文件
语法:
r := bufio.NewReader(file)
n, err := r.Read(buf)
参数:
- file:要读取的文件句柄
- buf:读取的数据存放的缓冲区
返回值: - n:读取到的长度
- err:读取失败,则返回错误信息
说明:
使用NewReader读取文件时,首先,我们需要打开文件,接着使用打开的文件返回的句柄当作函数参数传入NewReader,最后我们使用NewReader返回的reader对象调用Read读取文件,文件读取结束的标志是返回的n等于0,因此我们需要读取整个文件的内容,那么我们需要使用for循环不停的读取文件,直到n等于0. - 示例说明:
package main
import (
"bufio"
"fmt"
"io"
"os"
)
func main() {
fmt.Println("Open File Test")
fileName := "D:/01.代码目录/04.go项目/sensetime/metro-data-check.go"
file, err := os.Open(fileName)
if err != nil {
fmt.Println("Open file err:", err)
return
}
defer file.Close()
var chunk []byte
buf := make([]byte, 2048)
for {
r := bufio.NewReader(file)
n, err := r.Read(buf)
if err != nil && err != io.EOF {
fmt.Println("read buf fail", err)
return
}
if n == 0 {
break
}
chunk = append(chunk, buf[:n]...)
}
fmt.Println("File content is:\n", string(chunk))
}
2.4 ReadAll读取文件
语法:
func ReadAll(r io.Reader) ([]byte, error)
参数:
- []byte:读取到的数据
- error:读取失败,则返回错误信息
说明:
使用ReadAll读取文件时,首先我们需要打开文件,接着使用打开的文件返回的文件句柄当作函数参数传入ReadAll,ReadAll函数会将整个文件的内容一次性读取出来,如果读取出错,则返回error信息。
示例说明:
package main
import (
"fmt"
"io/ioutil"
"os"
)
func main() {
fmt.Println("Open File Test")
fileName := "D:/01.代码目录/04.go项目/sensetime/metro-data-check.go"
file, err := os.Open(fileName)
if err != nil {
fmt.Println("Open file err:", err)
return
}
defer file.Close()
fileContent, err := ioutil.ReadAll(file)
if err != nil {
fmt.Println("Read file err: ", err)
} else {
fmt.Println("File Content:\n", string(fileContent))
}
}