golang 读取文件的四种方式

本文介绍了使用Go语言进行文件读写的多种方法,并对比了不同读取方式的效率。包括直接利用ioutil读取文件、通过os包结合缓冲区读取、使用bufio.Reader逐块读取,以及利用ioutil.ReadAll进行读取等。同时,还提供了写文件的四种常见实现方式。

参考

Go实战--golang中读写文件的几种方式

读文件

读取的文件放在file/test:也就是file包下的test这个文件,里面写多一点文件

  • 读文件方式一:利用ioutil.ReadFile直接从文件读取到[]byte中
func Read0()  (string){
    f, err := ioutil.ReadFile("file/test")
    if err != nil {
        fmt.Println("read fail", err)
    }
    return string(f)
}

  • 读文件方式二:先从文件读取到file中,在从file读取到buf, buf在追加到最终的[]byte
func Read1()  (string){
    //获得一个file
    f, err := os.Open("file/test")
    if err != nil {
        fmt.Println("read fail")
        return ""
    }

    //把file读取到缓冲区中
    defer f.Close()
    var chunk []byte
    buf := make([]byte, 1024)

    for {
        //从file读取到buf中
        n, err := f.Read(buf)
        if err != nil && err != io.EOF{
            fmt.Println("read buf fail", err)
            return ""
        }
        //说明读取结束
        if n == 0 {
            break
        }
        //读取到最终的缓冲区中
        chunk = append(chunk, buf[:n]...)
    }

    return string(chunk)
    //fmt.Println(string(chunk))
}

  • 读文件方式三:先从文件读取到file, 在从file读取到Reader中,从Reader读取到buf, buf最终追加到[]byte
//先从文件读取到file, 在从file读取到Reader中,从Reader读取到buf, buf最终追加到[]byte,这个排第三
func Read2() (string) {
    fi, err := os.Open("file/test")
    if err != nil {
        panic(err)
    }
    defer fi.Close()

    r := bufio.NewReader(fi)
    var chunks []byte

    buf := make([]byte, 1024)

    for {
        n, err := r.Read(buf)
        if err != nil && err != io.EOF {
            panic(err)
        }
        if 0 == n {
            break
        }
        //fmt.Println(string(buf))
        chunks = append(chunks, buf...)
    }
    return string(chunks)
    //fmt.Println(string(chunks))
}

  • 读文件方式四:读取到file中,再利用ioutil将file直接读取到[]byte中
//读取到file中,再利用ioutil将file直接读取到[]byte中, 这是最优
func Read3()  (string){
    f, err := os.Open("file/test")
    if err != nil {
        fmt.Println("read file fail", err)
        return ""
    }
    defer f.Close()

    fd, err := ioutil.ReadAll(f)
    if err != nil {
        fmt.Println("read to fd fail", err)
        return ""
    }

    return string(fd)
}

读取速度比较

经过我的测试,这四种方式读的速度排名是:前者为优
方式四 > 方式一 > 方式三 > 方式四

写文件

  • 写文件方式一:使用 io.WriteString 写入文件
func Write0()  {
    fileName := "file/test1"
    strTest := "测试测试"

    var f *os.File
    var err error

    if CheckFileExist(fileName) {  //文件存在
        f, err = os.OpenFile(fileName, os.O_APPEND, 0666) //打开文件
        if err != nil{
            fmt.Println("file open fail", err)
            return
        }
    }else {  //文件不存在
        f, err = os.Create(fileName) //创建文件
        if err != nil {
            fmt.Println("file create fail")
            return
        }
    }
    //将文件写进去
    n, err1 := io.WriteString(f, strTest)
    if err1 != nil {
        fmt.Println("write error", err1)
        return
    }
    fmt.Println("写入的字节数是:", n)
}

  • 写文件方式二:使用 ioutil.WriteFile 写入文件
func Write1()  {
    fileName := "file/test2"
    strTest := "测试测试"
    var d = []byte(strTest)
    err := ioutil.WriteFile(fileName, d, 0666)
    if err != nil {
        fmt.Println("write fail")
    }
    fmt.Println("write success")
}

  • 写文件方式三:使用 File(Write,WriteString) 写入文件
func Write2()  {

    fileName := "file/test3"
    strTest := "测试测试"
    var d1 = []byte(strTest)

    f, err3 := os.Create(fileName) //创建文件
    if err3 != nil{
        fmt.Println("create file fail")
    }
    defer f.Close()
    n2, err3 := f.Write(d1) //写入文件(字节数组)

    fmt.Printf("写入 %d 个字节n", n2)
    n3, err3 := f.WriteString("writesn") //写入文件(字节数组)
    fmt.Printf("写入 %d 个字节n", n3)
    f.Sync()
}

  • 写文件方式四:使用 bufio.NewWriter 写入文件
func Write3()  {
    fileName := "file/test3"
    f, err3 := os.Create(fileName) //创建文件
    if err3 != nil{
        fmt.Println("create file fail")
    }
    w := bufio.NewWriter(f) //创建新的 Writer 对象
    n4, err3 := w.WriteString("bufferedn")
    fmt.Printf("写入 %d 个字节n", n4)
    w.Flush()
    f.Close()
}

  • 检查文件是否存在:
func CheckFileExist(fileName string) bool {
    _, err := os.Stat(fileName)
    if os.IsNotExist(err) {
        return false
    }
    return true
}

注意事项

  • 本文参考较大,如有侵权,请私信我处理,谢谢
  • 写文件的几种方式没有经过比较,如果有比较的话,也可以评论此篇文章告知我结果,谢谢

 

在Go语言中,有多种读取文件的方法: 1. **使用`os.Open`和`io/ioutil.ReadAll`一次性读取整个文件**:Go语言的标准库提供了`os`和`io/ioutil`等包,可使用`os.Open`打开文件,`io/ioutil.ReadAll`读取文件全部内容,示例代码如下: ```go package main import ( "fmt" "io/ioutil" "os" ) func main() { file, err := os.Open("your_file.txt") if err != nil { fmt.Println("Error opening file:", err) return } defer file.Close() data, err := ioutil.ReadAll(file) if err != nil { fmt.Println("Error reading file:", err) return } fmt.Println(string(data)) } ``` 这种方法适合读取较小的文件,能简单直接地获取文件全部内容[^1]。 2. **使用`os`包的相关方法**:操作文件的相关方法在`$GOROOT/src/os/file.go`文件中,主要函数有`OpenFile()`、`Open()`、`ReadFile()`等。 - `OpenFile()`:是较底层的打开文件方式,可灵活配置打开的文件,例如`Create()`方法就是调用该方法实现文件的新建。 - `Open()`:以只读的方式打开一个文件。 - `ReadFile()`:调用`Open()`方法打开文件,循环读取文件内容,返回值是字节数组。示例代码如下: ```go package main import ( "fmt" "os" ) func main() { data, err := os.ReadFile("your_file.txt") if err != nil { fmt.Println("Error reading file:", err) return } fmt.Println(string(data)) } ``` 这里使用`os.ReadFile`直接读取文件内容,若文件不存在会返回错误[^2]。 3. **分块读取文件**:一次性读取整个文件有时不利于内存管理,Golang 提供了分块读取文件的方法,而非整个或逐行读取,避免因一行大小太大导致逐行读取低效。示例代码如下: ```go package main import ( "bufio" "fmt" "io" "log" "os" ) func main() { f, err := os.Open("ilovego.txt") if err != nil { log.Fatal(err) } defer f.Close() reader := bufio.NewReader(f) buf := make([]byte, 16) for { // reading a file upto buffer n, err := reader.Read(buf) if err != nil { if err != io.EOF { log.Fatal(err) } break } fmt.Print(string(buf[0:n])) } fmt.Println() } ``` 该示例中使用`bufio.NewReader`创建一个带缓冲的读取器,每次读取 16 字节的数据,直到文件结束[^3]。
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值