温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

GoLang读取文件的方法有哪些

发布时间:2022-06-29 14:16:22 来源:亿速云 阅读:132 作者:iii 栏目:开发技术

这篇文章主要介绍“GoLang读取文件的方法有哪些”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“GoLang读取文件的方法有哪些”文章能帮助大家解决问题。

    一. 整个文件读入内存

    直接将数据直接读取入内存,是效率最高的一种方式,但此种方式,仅适用于小文件,对于大文件,则不适合,因为比较浪费内存

    1.直接指定文化名读取

    在 Go 1.16 开始,ioutil.ReadFile 就等价于 os.ReadFile,二者是完全一致的

    1.1使用os.ReadFile函数读取文件
    package main
    
    import (
        "fmt"
        "os"
    )
    
    func main() {
       //func ReadFile(name string) ([]byte, error) {}
        content, err := os.ReadFile("a.txt")
        if err != nil {
            panic(err)
        }
        fmt.Println(string(content))
    }

    1.2使用ioutil.ReadFile函数读取文件

    package main
    
    import (
        "io/ioutil"
        "fmt"
    )
    
    func main() {
        content, err := ioutil.ReadFile("a.txt")
        if err != nil {
            panic(err)
        }
        fmt.Println(string(content))
    }
    // As of Go 1.16, this function simply calls os.ReadFile.
    func ReadFile(filename string) ([]byte, error) {
        return os.ReadFile(filename)
    }

    2.先创建句柄再读取

    2.1使用os.OpenFile函数只读形式获取句柄
    package main
    
    import (
    "os"
    "io/ioutil"
    "fmt"
    )
    
    func main() {
        /*func Open(name string) (*File, error) {
    	return OpenFile(name, O_RDONLY, 0)
         }*/
         //Open是一个高级函数,是因为它是只读模式来打开文件
         /*也可以直接使用 os.OpenFile,只是要多加两个参数
           file, err := os.OpenFile("a.txt", os.O_RDONLY, 0)*/
        file, err := os.Open("a.txt")
        if err != nil {
            panic(err)
        }
        //func (f *File) Close() error {}
        defer file.Close()
        //func ReadAll(r io.Reader) ([]byte, error) {}
        content, err := ioutil.ReadAll(file)
        fmt.Println(string(content))
    }
    2.2代码讲解

    2.2.1os.File结构体

    type File struct {
    	*file // os specific
    }

    2.2.2os.OpenFile函数

    func OpenFile(name string, flag int, perm FileMode) (
        *File,   error) {}

    2.2.3io.Reader接口

    type Reader interface {
    	Read(p []byte) (n int, err error)
    }

    二.每次只读取一行

    一次性读取所有的数据,太耗费内存,因此可以指定每次只读取一行数据,方法有三种:

    (1)bufio.读行()

    (2)bufio.读取字节("\n")

    (3)bufio.ReadString(’\n’)

    在 bufio 的源码注释中,曾说道 bufio.ReadLine() 是低级库,不太适合普通用户使用,更推荐用户使用 bufio.ReadBytes和bufio.ReadString 去读取单行数据

    因此,这里不再介绍 bufio.读行()

    1.使用bufio.Reader结构体的ReadBytes方法读取字节数

    ReadBytes读取直到第一次遇到delim字节,返回一个包含已读取的数据和delim字节的切片。如果ReadBytes方法在读取到delim之前遇到了错误,它会返回在错误之前读取的数据以及该错误(一般是io.EOF)。当且仅当ReadBytes方法返回的切片不以delim结尾时,会返回一个非nil的错误

    package main
    
    import (
        "bufio"
        "fmt"
        "io"
        "os"
        "strings"
    )
    
    func main() {
        // 创建句柄
        fi, err := os.Open("christmas_apple.py")
        if err != nil {
            panic(err)
        }
        //func NewReader(rd io.Reader) *Reader {},返回的是bufio.Reader结构体
        r := bufio.NewReader(fi)// 创建 Reader
    
        for {
        //func (b *Reader) ReadBytes(delim byte) ([]byte, error) {}
            lineBytes, err := r.ReadBytes('\n')
            //去掉字符串首尾空白字符,返回字符串
            line := strings.TrimSpace(string(lineBytes))
            if err != nil && err != io.EOF {
                panic(err)
            }
            if err == io.EOF {
                break
            }
            fmt.Println(line)
        }
    }

    2.使用bufio.Reader结构体的ReadString方法读取字符串

    ReadString读取直到第一次遇到delim字节,返回一个包含已读取的数据和delim字节的字符串。如果ReadString方法在读取到delim之前遇到了错误,它会返回在错误之前读取的数据以及该错误(一般是io.EOF)。当且仅当ReadString方法返回的切片不以delim结尾时,会返回一个非nil的错误

    package main
    
    import (
        "bufio"
        "fmt"
        "io"
        "os"
        "strings"
    )
    
    func main() {
        // 创建句柄
        fi, err := os.Open("a.txt")
        if err != nil {
            panic(err)
        }
    
        // 创建 Reader
        r := bufio.NewReader(fi)
    
        for {
        //func (b *Reader) ReadString(delim byte) (string, error) {}
            line, err := r.ReadString('\n')
            line = strings.TrimSpace(line)
            if err != nil && err != io.EOF {
                panic(err)
            }
            if err == io.EOF {
                break
            }
            fmt.Println(line)
        }
    }

    3.代码讲解

    3.1bufio.Reader结构体
    type Reader struct {
    	buf          []byte
    	rd           io.Reader // reader provided by the client
    	r, w         int       // buf read and write positions
    	err          error
    	lastByte     int // last byte read for UnreadByte; -1 means invalid
    	lastRuneSize int // size of last rune read for UnreadRune; -1 means invalid
    }

    三.每次只读取固定字节数

    每次仅读取一行数据,可以解决内存占用过大的问题,但要注意的是,并不是所有的文件都有换行符 \n;
    因此对于一些不换行的大文件来说,还得再想想其他办法

    1.使用os库

    通用的做法是:

    先创建一个文件句柄,可以使用 os.Open 或者 os.OpenFile;

    然后 bufio.NewReader 创建一个 Reader;

    然后在 for 循环里调用 Reader 的 Read 函数,每次仅读取固定字节数量的数据

    Read方法读取数据写入p;本方法返回写入p的字节数;本方法一次调用最多会调用下层Reader接口一次Read方法,因此返回值n可能小于len§;读取到达结尾时,返回值n将为0而err将为io.EOF

    package main
    
    import (
        "bufio"
        "fmt"
        "io"
        "os"
    )
    
    func main() {
        // 创建句柄
        fi, err := os.Open("a.txt")
        if err != nil {
            panic(err)
        }
    
        // 创建 Reader
        r := bufio.NewReader(fi)
    
        // 每次读取 1024 个字节
        buf := make([]byte, 1024)
        for {
            //func (b *Reader) Read(p []byte) (n int, err error) {}
            n, err := r.Read(buf)
            if err != nil && err != io.EOF {
                panic(err)
            }
    
            if n == 0 {
                break
            }
            fmt.Println(string(buf[:n]))
        }
    }

    2.使用 syscall库

    os 库本质上也是调用 syscall 库,但由于 syscall 过于底层,如非特殊需要,一般不会使用 syscall;

    本篇为了内容的完整度,这里也使用 syscall 来举个例子;

    本例中,会每次读取 100 字节的数据,并发送到通道中,由另外一个协程进行读取并打印出来

    package main
    
    import (
        "fmt"
        "sync"
        "syscall"
    )
    
    func main() {
        fd, err := syscall.Open("christmas_apple.py", syscall.O_RDONLY, 0)
        if err != nil {
            fmt.Println("Failed on open: ", err)
        }
        defer syscall.Close(fd)
    
        var wg sync.WaitGroup
        wg.Add(2)
        dataChan := make(chan []byte)
        go func() {
            wg.Done()
            for {
                data := make([]byte, 100)
                n, _ := syscall.Read(fd, data)
                if n == 0 {
                    break
                }
                dataChan <- data
            }
            close(dataChan)
        }()
    
        go func() {
            defer wg.Done()
            for {
                select {
                case data, ok := <-dataChan:
                    if !ok {
                        return
                    }
    
                    fmt.Printf(string(data))
                default:
    
                }
            }
        }()
        wg.Wait()
    }

    关于“GoLang读取文件的方法有哪些”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识,可以关注亿速云行业资讯频道,小编每天都会为大家更新不同的知识点。

    向AI问一下细节

    免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

    AI