推荐使用 io.copy 配合 os.Open 和 os.Create 实现高效安全的文件拷贝;需 defer 关闭文件句柄;若保留权限和时间戳,应调用 os.Stat 获取信息后用 Chmod/Chtimes 设置;小文件可用 ReadFile/WriteFile 简化操作但不适用于大文件。

用 go 实现文件拷贝,核心思路是:打开源文件读取内容,再写入目标文件。推荐使用 io.Copy,它内部做了缓冲优化,比手动读写更高效、更安全。
基础方式:io.Copy + os.Open / os.Create
这是最常用也最推荐的方式,适合大多数场景(包括大文件):
- 用
os.Open打开源文件(只读) - 用
os.Create创建或截断目标文件(可写) - 调用
io.Copy(dst, src)完成流式拷贝 - 务必关闭两个文件句柄,建议用
defer
示例代码:
package main import ( "io" "log" "os" ) func copyFile(src, dst string) error { source, err := os.Open(src) if err != nil { return err } defer source.Close() dest, err := os.Create(dst) if err != nil { return err } defer dest.Close() _, err = io.Copy(dest, source) return err } func main() { if err := copyFile("a.txt", "b.txt"); err != nil { log.Fatal(err) } }
带权限和时间戳的拷贝(类似 cp -p)
如果需要保留源文件的权限(mode)、修改时间(ModTime),需额外获取并设置:
立即学习“go语言免费学习笔记(深入)”;
- 用
os.Stat获取源文件os.FileInfo - 创建目标文件后,用
dest.Chmod和dest.Chtimes设置属性
注意:os.Create 默认权限是 0666,会被 umask 限制;建议先用 os.OpenFile 配合 os.O_CREATE|os.O_WRONLY,再显式 Chmod。
小文件直接读全部再写入(不推荐用于大文件)
仅适用于确定很小的文件(如配置文件),代码简洁但会占用内存:
示例:
data, err := os.ReadFile("src.txt") if err != nil { log.Fatal(err) } err = os.WriteFile("dst.txt", data, 0644) if err != nil { log.Fatal(err) }
错误处理与健壮性建议
生产环境建议增强以下细节:
- 检查源文件是否存在且可读:
os.IsNotExist(err) - 确保目标目录存在,缺失时用
os.MkdirAll(filepath.Dir(dst), 0755) - 拷贝前可先校验源文件大小(避免空文件误拷)
- 若需原子性,可先写入临时文件,成功后再
os.Rename替换目标