Go语言文件操作

文章引用自

打开和关闭文件

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

package main

import (
        "fmt"
        "os"
)

func main() {
        // 只读方式打开当前目录下的main.go文件
        file, err := os.Open("./main.go") // 接受文件句柄file以及error错误信息
        if err != nil {
                fmt.Println("open file failed!, err:", err)
                return
        }
        // 关闭文件
        file.Close()
}

  

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

读取文件

file.Read()

基本使用

Read方法定义如下:

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

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

func main() {
        // 只读方式打开当前目录下的main.go文件
        file, err := os.Open("./main.go")
        if err != nil {
                fmt.Println("open file failed!, err:", err)
                return
        }
        defer file.Close()
        // 使用Read方法读取数据
        var tmp = make([]byte, 128) // 指定一次性读取的字节 // 传入切片的大小就是能读取到的最大长度
        n, err := file.Read(tmp)  // n为读取的字节数
        if err == io.EOF {
                fmt.Println("文件读完了")
                return
        }
        if err != nil {
                fmt.Println("read file failed, err:", err)
                return
        }
        fmt.Printf("读取了%d字节数据\n", n)
        fmt.Println(string(tmp[:n]))
}

  

循环读取

使用for循环读取文件中的所有数据。

func main() {
        // 只读方式打开当前目录下的main.go文件
        file, err := os.Open("./main.go")
        if err != nil {
                fmt.Println("open file failed!, err:", err)
                return
        }
        defer file.Close()
        // 循环读取文件
        var content []byte
        var tmp = make([]byte, 128)
        for {
                n, err := file.Read(tmp)
                if err == io.EOF {
                        fmt.Println("文件读完了")
                        break
                }
                if err != nil {
                        fmt.Println("read file failed, err:", err)
                        return
                }
                content = append(content, tmp[:n]...)
        }
        fmt.Println(string(content))
}

  

bufio读取文件

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

主要是按照指定字符结束读取,解决了file.Read()方法中指定字符读到末尾乱码

package main

import (
        "bufio"
        "fmt"
        "io"
        "os"
)

// bufio按行读取示例
func main() {
        file, err := os.Open("./main.go")
        if err != nil {
                fmt.Println("open file failed, err:", err)
                return
        }
        defer file.Close()
        reader := bufio.NewReader(file)  // 创建bufio对象 传入file句柄
        for {
                line, err := reader.ReadString('\n') //注意是字符 // 指定字符  // 还有其他方法这里列出ReadString 还有ReadByte ReadLine 等等...
                if err == io.EOF {
                        fmt.Println("文件读完了")
                        break
                }
                if err != nil {
                        fmt.Println("read file failed, err:", err)
                        return
                }
                fmt.Print(line)
        }
}

  

ioutil读取整个文件

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

package main

import (
        "fmt"
        "io/ioutil"
)

// ioutil.ReadFile读取整个文件
func main() {
        content, err := ioutil.ReadFile("./main.go")  // 创建ioutil对象
        if err != nil {
                fmt.Println("read file failed, err:", err)
                return
        }
        fmt.Println(string(content))
}

  

文件写入操作

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

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

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

bufio.NewWriter

  // 按指定字符终止去写

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

  

ioutil.WriteFile

  // 整篇文章可以直接写入

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

句柄的其他用法

  // 用法与os.Stat类似

fileinfo, err := fileHand.Stat() // fileHand值拿到句柄
    if err != nil {
        panic(err)
    }
    fmt.Println(fileinfo.Name())    //获取文件名
    fmt.Println(fileinfo.IsDir())   //判断是否是目录,返回bool类型
    fmt.Println(fileinfo.ModTime()) //获取文件修改时间
    fmt.Println(fileinfo.Mode())  //获取文件权限
    fmt.Println(fileinfo.Size()) //获取文件大小
    fmt.Println(fileinfo.Sys())  // 获取文件相关信息
}

  

os.Stat() 方法与功能

fileinfo, err := os.Stat(`C:\Users\Administrator\Desktop\UninstallTool.zip`)
    if err != nil {
        panic(err)
    }
    fmt.Println(fileinfo.Name())    //获取文件名
    fmt.Println(fileinfo.IsDir())   //判断是否是目录,返回bool类型
    fmt.Println(fileinfo.ModTime()) //获取文件修改时间
    fmt.Println(fileinfo.Mode())  //获取文件权限
    fmt.Println(fileinfo.Size()) //获取文件大小
    fmt.Println(fileinfo.Sys())  // 获取文件相关信息
}

 

练习

copyFile

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

// CopyFile 拷贝文件函数
func CopyFile(dstName, srcName string) (written int64, err error) {
        // 以读方式打开源文件
        src, err := os.Open(srcName)
        if err != nil {
                fmt.Printf("open %s failed, err:%v.\n", srcName, err)
                return
        }
        defer src.Close()
        // 以写|创建的方式打开目标文件
        dst, err := os.OpenFile(dstName, os.O_WRONLY|os.O_CREATE, 0644)
        if err != nil {
                fmt.Printf("open %s failed, err:%v.\n", dstName, err)
                return
        }
        defer dst.Close()
        return io.Copy(dst, src) //调用io.Copy()拷贝内容
}
func main() {
        _, err := CopyFile("dst.txt", "src.txt")
        if err != nil {
                fmt.Println("copy file failed, err:", err)
                return
        }
        fmt.Println("copy done!")
}

  

实现一个cat命令

使用文件操作相关知识,模拟实现linux平台cat命令的功能。

package main

import (
        "bufio"
        "flag"
        "fmt"
        "io"
        "os"
)

// cat命令实现
func cat(r *bufio.Reader) {
        for {
                buf, err := r.ReadBytes('\n') //注意是字符
                if err == io.EOF {
                        break
                }
                fmt.Fprintf(os.Stdout, "%s", buf)
        }
}

func main() {
        flag.Parse() // 解析命令行参数
        if flag.NArg() == 0 {
                // 如果没有参数默认从标准输入读取内容
                cat(bufio.NewReader(os.Stdin))
        }
        // 依次读取每个指定文件的内容并打印到终端
        for i := 0; i < flag.NArg(); i++ {
                f, err := os.Open(flag.Arg(i))
                if err != nil {
                        fmt.Fprintf(os.Stdout, "reading from %s failed, err:%v\n", flag.Arg(i), err)
                        continue
                }
                cat(bufio.NewReader(f))
        }
}

  

打开和关闭文件

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

package main

import (
        "fmt"
        "os"
)

func main() {
        // 只读方式打开当前目录下的main.go文件
        file, err := os.Open("./main.go") // 接受文件句柄file以及error错误信息
        if err != nil {
                fmt.Println("open file failed!, err:", err)
                return
        }
        // 关闭文件
        file.Close()
}

  

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

读取文件

file.Read()

基本使用

Read方法定义如下:

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

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

func main() {
        // 只读方式打开当前目录下的main.go文件
        file, err := os.Open("./main.go")
        if err != nil {
                fmt.Println("open file failed!, err:", err)
                return
        }
        defer file.Close()
        // 使用Read方法读取数据
        var tmp = make([]byte, 128) // 指定一次性读取的字节 // 传入切片的大小就是能读取到的最大长度
        n, err := file.Read(tmp)  // n为读取的字节数
        if err == io.EOF {
                fmt.Println("文件读完了")
                return
        }
        if err != nil {
                fmt.Println("read file failed, err:", err)
                return
        }
        fmt.Printf("读取了%d字节数据\n", n)
        fmt.Println(string(tmp[:n]))
}

  

循环读取

使用for循环读取文件中的所有数据。

func main() {
        // 只读方式打开当前目录下的main.go文件
        file, err := os.Open("./main.go")
        if err != nil {
                fmt.Println("open file failed!, err:", err)
                return
        }
        defer file.Close()
        // 循环读取文件
        var content []byte
        var tmp = make([]byte, 128)
        for {
                n, err := file.Read(tmp)
                if err == io.EOF {
                        fmt.Println("文件读完了")
                        break
                }
                if err != nil {
                        fmt.Println("read file failed, err:", err)
                        return
                }
                content = append(content, tmp[:n]...)
        }
        fmt.Println(string(content))
}

  

bufio读取文件

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

主要是按照指定字符结束读取,解决了file.Read()方法中指定字符读到末尾乱码

package main

import (
        "bufio"
        "fmt"
        "io"
        "os"
)

// bufio按行读取示例
func main() {
        file, err := os.Open("./main.go")
        if err != nil {
                fmt.Println("open file failed, err:", err)
                return
        }
        defer file.Close()
        reader := bufio.NewReader(file)  // 创建bufio对象 传入file句柄
        for {
                line, err := reader.ReadString('\n') //注意是字符 // 指定字符  // 还有其他方法这里列出ReadString 还有ReadByte ReadLine 等等...
                if err == io.EOF {
                        fmt.Println("文件读完了")
                        break
                }
                if err != nil {
                        fmt.Println("read file failed, err:", err)
                        return
                }
                fmt.Print(line)
        }
}

  

ioutil读取整个文件

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

package main

import (
        "fmt"
        "io/ioutil"
)

// ioutil.ReadFile读取整个文件
func main() {
        content, err := ioutil.ReadFile("./main.go")  // 创建ioutil对象
        if err != nil {
                fmt.Println("read file failed, err:", err)
                return
        }
        fmt.Println(string(content))
}

  

文件写入操作

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

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

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

bufio.NewWriter

  // 按指定字符终止去写

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

  

ioutil.WriteFile

  // 整篇文章可以直接写入

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

句柄的其他用法

  // 用法与os.Stat类似

fileinfo, err := fileHand.Stat() // fileHand值拿到句柄
    if err != nil {
        panic(err)
    }
    fmt.Println(fileinfo.Name())    //获取文件名
    fmt.Println(fileinfo.IsDir())   //判断是否是目录,返回bool类型
    fmt.Println(fileinfo.ModTime()) //获取文件修改时间
    fmt.Println(fileinfo.Mode())  //获取文件权限
    fmt.Println(fileinfo.Size()) //获取文件大小
    fmt.Println(fileinfo.Sys())  // 获取文件相关信息
}

  

os.Stat() 方法与功能

fileinfo, err := os.Stat(`C:\Users\Administrator\Desktop\UninstallTool.zip`)
    if err != nil {
        panic(err)
    }
    fmt.Println(fileinfo.Name())    //获取文件名
    fmt.Println(fileinfo.IsDir())   //判断是否是目录,返回bool类型
    fmt.Println(fileinfo.ModTime()) //获取文件修改时间
    fmt.Println(fileinfo.Mode())  //获取文件权限
    fmt.Println(fileinfo.Size()) //获取文件大小
    fmt.Println(fileinfo.Sys())  // 获取文件相关信息
}

 

练习

copyFile

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

// CopyFile 拷贝文件函数
func CopyFile(dstName, srcName string) (written int64, err error) {
        // 以读方式打开源文件
        src, err := os.Open(srcName)
        if err != nil {
                fmt.Printf("open %s failed, err:%v.\n", srcName, err)
                return
        }
        defer src.Close()
        // 以写|创建的方式打开目标文件
        dst, err := os.OpenFile(dstName, os.O_WRONLY|os.O_CREATE, 0644)
        if err != nil {
                fmt.Printf("open %s failed, err:%v.\n", dstName, err)
                return
        }
        defer dst.Close()
        return io.Copy(dst, src) //调用io.Copy()拷贝内容
}
func main() {
        _, err := CopyFile("dst.txt", "src.txt")
        if err != nil {
                fmt.Println("copy file failed, err:", err)
                return
        }
        fmt.Println("copy done!")
}

  

实现一个cat命令

使用文件操作相关知识,模拟实现linux平台cat命令的功能。

package main

import (
        "bufio"
        "flag"
        "fmt"
        "io"
        "os"
)

// cat命令实现
func cat(r *bufio.Reader) {
        for {
                buf, err := r.ReadBytes('\n') //注意是字符
                if err == io.EOF {
                        break
                }
                fmt.Fprintf(os.Stdout, "%s", buf)
        }
}

func main() {
        flag.Parse() // 解析命令行参数
        if flag.NArg() == 0 {
                // 如果没有参数默认从标准输入读取内容
                cat(bufio.NewReader(os.Stdin))
        }
        // 依次读取每个指定文件的内容并打印到终端
        for i := 0; i < flag.NArg(); i++ {
                f, err := os.Open(flag.Arg(i))
                if err != nil {
                        fmt.Fprintf(os.Stdout, "reading from %s failed, err:%v\n", flag.Arg(i), err)
                        continue
                }
                cat(bufio.NewReader(f))
        }
}