我有以下功能将文件(实际上是io.Reader)复制到目标字符串位置。但是,似乎只有部分文件被实际复制,导致文件损坏。我做错了什么?
func CopyFile(in io.Reader, dst string) (err error) {
// Does file already exist? Skip
if _, err := os.Stat(dst); err == nil {
return nil
}
err = nil
out, err := os.Create(dst)
if err != nil {
fmt.Println("Error creating file", err)
return
}
defer func() {
cerr := out.Close()
if err == nil {
err = cerr
}
}()
var bytes int64
if bytes, err = io.Copy(out, in); err != nil {
fmt.Println("io.Copy error")
return
}
fmt.Println(bytes)
err = out.Sync()
return
}
我正在使用filepath.Walk(dir, visit)
方法来处理目录中的文件。
// Process each matching file on our walk down the filesystem
func visit(path string, f os.FileInfo, err error) error {
if reader, err := os.Open(path); err == nil {
defer reader.Close()
// http://golang.org/pkg/os/#FileInfo
statinfo, err := reader.Stat()
if err != nil {
fmt.Println(err)
return nil
}
fmt.Println()
fmt.Println(statinfo.Size())
// Directory exists and is writable
err = CopyFile(reader, "/tmp/foo/"+f.Name())
if err != nil {
fmt.Println(err)
}
} else {
fmt.Println("Impossible to open the file:", err)
}
}
当前closest question我可以接受建议使用硬/软链接的答案,如果文件已经存在则不会中止。
答案 0 :(得分:6)
package main
import (
"fmt"
"io"
"os"
)
func main() {
srcFile, err := os.Open("test.txt")
check(err)
defer srcFile.Close()
destFile, err := os.Create("test_copy.txt") // creates if file doesn't exist
check(err)
defer destFile.Close()
_, err = io.Copy(destFile, srcFile) // check first var for number of bytes copied
check(err)
err = destFile.Sync()
check(err)
}
func check(err error) {
if err != nil {
fmt.Println("Error : %s", err.Error())
os.Exit(1)
}
}
此代码适用于我。请检查使用io.Copy
的返回值复制的字节数。
答案 1 :(得分:0)
复制golang的最简单方法是 - http://golang.org/pkg/io/#Copy
答案 2 :(得分:0)
另一个选项是ReadFrom
:
package main
import "os"
func copyFile(in, out string) (int64, error) {
i, e := os.Open(in)
if e != nil { return 0, e }
defer i.Close()
o, e := os.Create(out)
if e != nil { return 0, e }
defer o.Close()
return o.ReadFrom(i)
}
func main() {
_, e := copyFile("in.txt", "out.txt")
if e != nil {
panic(e)
}
}