当前位置:   article > 正文

go语言文件操作

go语言文件操作

打开和关闭文件

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. // 关闭文件
  14. file.Close()
  15. }

为了防止文件忘记关闭,我们通常使用defer注册文件关闭语句。

读取文件

file.Read()

基本使用

Read方法定义如下:

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方法读取数据
  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循环读取文件中的所有数据。

  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. }

bufio读取文件

bufio是在file的基础上封装了一层API,支持更多的功能。

  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. }

读取整个文件

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. }

文件写入操作

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。

Write和WriteString

  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. }

bufio.NewWriter

  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. }

ioutil.WriteFile

  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. }

文件拷贝

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

  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()拷贝内容
  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、r.ReadSlice('\n')
最原始的方式,如果超出了 buf 长度会正常返回,且 err = ErrBufferFull

2、r.ReadString('\n')
对 ReadSlice的简单调用,只是返回的是字符串。

3、r.ReadLine()
也是对 ReadSlice的调用,只是它会删掉末尾的换行符,多此一举,低效。

4、r.ReadBytes('\n')
也是对 ReadSlice的调用,不同的是,当 err = ErrBufferFull 时,它会扩展 buf 的大小,并多次调用 ReadSlice,直到这一行被读完,相对智能一些,推荐使用。

  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. // 退出之前将已读到的内容输出
  15. fmt.Fprintf(os.Stdout, "%s", buf)
  16. break
  17. }
  18. fmt.Fprintf(os.Stdout, "%s", buf)
  19. }
  20. }
  21. func main() {
  22. flag.Parse() // 解析命令行参数
  23. if flag.NArg() == 0 {
  24. // 如果没有参数默认从标准输入读取内容
  25. cat(bufio.NewReader(os.Stdin))
  26. }
  27. // 依次读取每个指定文件的内容并打印到终端
  28. for i := 0; i < flag.NArg(); i++ {
  29. f, err := os.Open(flag.Arg(i))
  30. if err != nil {
  31. fmt.Fprintf(os.Stdout, "reading from %s failed, err:%v\n", flag.Arg(i), err)
  32. continue
  33. }
  34. cat(bufio.NewReader(f))
  35. }
  36. }
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/article/detail/42769
推荐阅读
相关标签
  

闽ICP备14008679号