C++怎么复制文件 C++中std::copy文件流操作【总结】

2次阅读

C++怎么复制文件 C++中std::copy文件流操作【总结】

std::copy 配合文件流不能直接复制文件

它只能逐字节搬运数据,但不会自动创建目标文件、不处理二进制/文本模式差异、也不保证原子性或错误恢复。很多人一上来就写 std::copy(ifs, ofs),结果目标文件为空、乱码,或者程序崩溃。

  • 必须显式以 std::ios::binary 模式打开输入和输出流,否则 windows 下换行符会被误转(rnn
  • std::copy 本身不检查 failbitbadbit,出错时可能静默失败
  • 目标路径的父目录不存在时,std::ofstream 构造会失败,不会自动创建目录

用 std::ifstream + std::ofstream + std::copy 最简可行方案

这是最贴近“用标准库完成复制”的做法,适合小到中等大小文件(

std::ifstream src("a.txt", std::ios::binary); std::ofstream dst("b.txt", std::ios::binary); if (!src || !dst) {     // 处理打开失败:检查路径、权限、磁盘空间 } std::copy(std::istreambuf_iterator<char>(src),           std::istreambuf_iterator<char>(),           std::ostreambuf_iterator<char>(dst));

  • 务必检查 src.good()dst.good(),而不仅是构造是否成功
  • 不要用 std::istream_iterator(跳过空白),它会破坏二进制内容;必须用 std::istreambuf_iterator
  • 复制后建议调用 dst.flush() 并检查 dst.good(),确认所有字节真正写入磁盘

大文件或需健壮性的场景别硬刚 std::copy

超过几十 MB 的文件,std::copy 配合迭代器会频繁调用底层 read()/write(),性能差,且无法控制缓冲区大小。更关键的是:它不提供进度、中断恢复、权限继承、符号链接处理等能力。

  • Linux/macOS 推荐直接调用 copy_filec++17):std::filesystem::copy_file("a", "b", std::filesystem::copy_options::overwrite_existing)
  • Windows 上若用 C++17,std::filesystem::copy_file 同样可用,且会正确处理 ACL 和时间戳
  • 若无法升级到 C++17,老项目建议封装 read()/write() 系统调用,手动分配 64KB 缓冲区,比 std::istreambuf_iterator 快 3–5 倍

std::filesystem::copy_file 的坑比想象中多

它看似开箱即用,但几个默认行为容易踩雷:

立即学习C++免费学习笔记(深入)”;

  • 默认不覆盖目标文件 —— 错误是 std::filesystem::filesystem_error,异常信息里含 File exists,不是 errno
  • 不复制文件权限(mode)除非显式加 std::filesystem::copy_options::copy_symlinks(名字有误导,实际控制权限复制)
  • 跨分区复制时,它内部仍是“读+写”,不是 rename(),所以失败后可能留下半截文件
  • 某些旧版 libstdc++(GCC std::filesystem::exists() 验证

真正安全的复制往往得组合判断:先 exists(),再 is_regular_file(),再选 copy_options,最后捕获 filesystem_error 分类处理。

text=ZqhQzanResources