在Go中向文件追加内容

125

所以我可以像这样从本地文件读取:

data, error := ioutil.ReadFile(name)

我可以写入本地文件

ioutil.WriteFile(filename, content, permission)

但是我该如何向文件中追加内容?是否有内置的方法?


4
没错,你理解得很对。ioutil 函数只是为了方便处理常见任务而设计的。如果你想要更多的控制权,可以看一下 os 包。 - Evan Shaw
7个回答

204

这个答案在Go1中有效:

f, err := os.OpenFile(filename, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
if err != nil {
    panic(err)
}

defer f.Close()

if _, err = f.WriteString(text); err != nil {
    panic(err)
}

3
为什么不使用0666文件模式?我认为这是更好的默认值。 - Dmitri Shuralyov
1
@SridharRatnakumar:请查看另一条评论和_man umask_。使用典型的umask 022,您将获得典型的权限:0666&〜022 = 0644 = rw-r--r-- - akavel
57
如果文件可能不存在,包含os.O_CREATE选项会很好。 - bugloaf
4
也许这是一个有点愚蠢的问题,但为什么需要同时使用 os.O_APPENDos.O_WRONLY?这肯定能起作用,但为什么两者都要用? - Dusan Gligoric
os.O_APPEND 是用于控制行为的,以追加模式运行,因此不需要维护文件指针。 os.O_WRONLY 用于声明文件模式,即读、写或两者都有,在这种情况下是写入模式。 - Cyberience
我知道文档中写着“必须指定 O_RDONLY、O_WRONLY 或 O_RDWR 中的一个”,但是 O_APPEND 不就可以隐含 O_WRONLY 吗?我想不出有哪种情况下你会使用 O_RDONLY|O_APPEND(而且只有一种牵强附会的用例可以使用 O_RDWR|O_APPEND)。你知道,os.Append 看起来像是标准库中合理的补充,与其他包装器 os.Openos.Create 并列。 - ardnew

73

Go文档有一个完美的例子

package main

import (
    "log"
    "os"
)

func main() {
    // If the file doesn't exist, create it, or append to the file
    f, err := os.OpenFile("access.log", os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
    if err != nil {
        log.Fatal(err)
    }
    if _, err := f.Write([]byte("appended some data\n")); err != nil {
        log.Fatal(err)
    }
    if err := f.Close(); err != nil {
        log.Fatal(err)
    }
}

32

搞定了

更多信息

f, err := os.OpenFile(filename, os.O_APPEND|os.O_WRONLY, 0644) 

n, err := f.WriteString(text) 

f.Close()

8
你应该同时使用os.O_WRONLY。 - zhaozhi
默认情况下,您会获得一个只读文件描述符。 - Han Qiu

8

我会使用fmt.Fprintf,因为它接受一个writer作为参数。而连接或文件也可被视为writer,可以轻松地以字符串形式写入。

f, err := os.OpenFile(filename, os.O_APPEND|os.O_WRONLY, 0600)
if err != nil {
    panic(err)
}

defer f.Close()
fmt.Fprintf(f, "%s", text)

I hope this help!

Javier,


6

如果你也想创建文件

f, err := os.OpenFile(filename, os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0600)


5
f, err := os.OpenFile(filename, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0644)
if err != nil {
    panic(err)
}

defer f.Close()

if _, err = f.WriteString(text); err != nil {
    panic(err)
}

通过向os.OpenFile函数添加标志,对golang网站提供的代码进行了轻微更改。默认情况下,该函数只允许读取文件而不允许编辑等其他功能。

@Mitar,你具体是什么意思,因为我正在使用不同的函数。但是,如果您问的是特定的附加方式,我会指向os.OpenFile函数,该函数可以接受标志以告诉文件如何处理,例如,您可以使用此标志创建所需的文件,如果它不存在:os.O_CREATE,或者在这种情况下,您可以使用os.O_APPEND标志进行附加。ps.您可以同时使用其中的几个。 - paradox earthling
你说你对得分最高的评论进行了轻微更改。但是我从被接受的答案中没有看到任何变化。具体是什么更改呢? - Mitar
只有 chmod 不同,设置为 644? - Mitar
@Mitar 抱歉之前没听懂你的意思,但是默认情况下,您只能读取文件,因此您需要指定该文件的权限。 - paradox earthling
抱歉,我的表述也不够清晰。我是说你应该更新你的答案,并在其中解释你所做的更改。 - Mitar
显示剩余2条评论

1

假设您想将文件current的内容添加到文件all中,则以下代码可用:

func updateTrx() {
    var err error
    var f *os.File

    // If the file doesn't exist, create it, or append to the file
    if f, err = os.OpenFile("all.csv", os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644); err != nil {
        log.Fatal(err)
    }
    defer func() {
        if err := f.Close(); err != nil {
            log.Fatal("error", err)
        }
    }()

    var current *os.File
    if current, err = os.OpenFile("current.csv", os.O_RDONLY, 0); err != nil {
        log.Fatal("error", err)
    }

    defer func() {
        if err := current.Close(); err != nil {
            log.Fatal("error", err)
        }
    }()

    if fileBytes, err := ioutil.ReadAll(current); err != nil {
        log.Fatal("error", err)
    } else {
        if _, err := f.Write([]byte(fileBytes)); err != nil {
            log.Fatal(err)
        }
    }
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接