当前位置: 首页 > news >正文

现代C++ 文件系统库

一、std::filesystem 的前世今生

C++11 之前,文件系统操作依赖于平台特定的 API(如 Windows 的CreateFile或 POSIX 的open),缺乏统一接口。C++17 正式将std::filesystem纳入标准库,该库最初由 Boost.Filesystem 演化而来,提供了跨平台的文件系统操作能力。

核心优势

  • 跨平台兼容性:一次编写,支持 Windows、Linux、macOS 等主流平台
  • RAII 设计:通过pathdirectory_iterator等类实现资源自动管理
  • 性能优化:底层封装系统 API,减少不必要的开销
// 编译时需添加编译选项:-std=c++17
#include <iostream>
#include <filesystem>namespace fs = std::filesystem;int main() {std::cout << "C++ filesystem version: " << fs::version() << std::endl;return 0;
}
二、路径操作:从字符串到 path 对象的蜕变
1. 路径构造与标准化

传统 C++ 中使用字符串拼接路径容易出错(如反斜杠转义),fs::path提供了安全的路径处理方式:

void path_operations() {// 自动适配平台分隔符fs::path p1 = "data/images/logo.png";fs::path p2 = "/usr/local/include/c++/11";// 标准化路径(消除.和..)fs::path p3 = "/home/user/../documents/./project";std::cout << "Normalized: " << p3.normalize() << std::endl; // 输出:/home/documents/project// 跨平台路径拼接fs::path base = "/app";fs::path sub = "data/logs";std::cout << "Combined: " << base / sub << std::endl; // 输出:/app/data/logs
}
2. 路径组件分解
void path_components() {fs::path p = "/user/documents/report.pdf";std::cout << "Filename: " << p.filename() << std::endl;       // report.pdfstd::cout << "Stem: " << p.stem() << std::endl;           // reportstd::cout << "Extension: " << p.extension() << std::endl;   // .pdfstd::cout << "Parent path: " << p.parent_path() << std::endl; // /user/documents// 遍历路径层级std::cout << "Path components: ";for (const auto& component : p) {std::cout << component << "/"; // 输出:user/documents/report.pdf/}
}
三、文件与目录操作:从属性查询到内容修改
1. 文件属性查询
void file_attributes() {fs::path file = "example.txt";// 基础属性if (fs::exists(file)) {std::cout << "File size: " << fs::file_size(file) << " bytes" << std::endl;std::cout << "Last write time: " << fs::last_write_time(file) << std::endl;// 类型判断if (fs::is_regular_file(file)) {std::cout << "It's a regular file" << std::endl;} else if (fs::is_directory(file)) {std::cout << "It's a directory" << std::endl;}}
}
2. 目录遍历与文件操作
void directory_operations() {fs::path dir = "src";// 遍历目录(C++17范围for)std::cout << "Files in " << dir << ":\n";for (const auto& entry : fs::directory_iterator(dir)) {std::cout << "  - " << entry.path() << std::endl;// 重命名文件if (entry.path().extension() == ".cpp") {fs::path new_name = entry.path().stem() + "_new" + entry.path().extension();fs::rename(entry.path(), new_name);}}// 创建多级目录fs::create_directories("build/intermediate/output");// 复制文件fs::copy_file("main.cpp", "backup/main.cpp.bak");// 删除文件(谨慎使用)// fs::remove("temp.txt");
}
四、文件搜索与内容统计
1. 递归搜索特定类型文件
void search_files(const fs::path& dir, const std::string& ext, std::vector<fs::path>& results) {try {for (const auto& entry : fs::directory_iterator(dir)) {if (fs::is_directory(entry)) {search_files(entry.path(), ext, results); // 递归搜索子目录} else if (entry.path().extension() == ext) {results.push_back(entry.path());}}} catch (const fs::filesystem_error& e) {std::cerr << "Error: " << e.what() << std::endl;}
}int main() {std::vector<fs::path> cpp_files;search_files("project", ".cpp", cpp_files);std::cout << "Found " << cpp_files.size() << " C++ files:\n";for (const auto& p : cpp_files) {std::cout << "  - " << p << std::endl;}return 0;
}
2. 目录大小统计
uintmax_t calculate_directory_size(const fs::path& dir) {uintmax_t size = 0;for (const auto& entry : fs::recursive_directory_iterator(dir)) {if (fs::is_regular_file(entry)) {size += fs::file_size(entry);}}return size;
}void print_size_human_readable(uintmax_t bytes) {if (bytes < 1024) {std::cout << bytes << " B";} else if (bytes < 1024 * 1024) {std::cout << bytes / 1024.0 << " KB";} else {std::cout << bytes / (1024.0 * 1024) << " MB";}
}
五、性能优化
  1. 避免不必要的递归
    • 使用recursive_directory_iterator时,可通过skip_directory()跳过特定目录
fs::recursive_directory_iterator it(dir);
it.skip_directory(); // 跳过当前目录
  1. 缓存路径状态
    • 频繁查询属性时,使用fs::file_status缓存结果
fs::file_status status = fs::status(file);
if (fs::is_regular_file(status)) { /*...*/ }

  1. 错误处理策略
    • 操作文件系统时,始终捕获fs::filesystem_error异常
try {fs::remove_all("temp");
} catch (const fs::filesystem_error& e) {std::cerr << "Remove error: " << e.path1() << " - " << e.what() << std::endl;
}
六、跨平台注意事项
  1. 路径分隔符

    • Windows 使用反斜杠\,POSIX 使用正斜杠/fs::path会自动适配
  2. 权限管理

    • Windows 和 POSIX 的权限模型不同,需注意fs::permissions的平台差异
  3. 符号链接处理

    • Linux 中符号链接常见,可通过fs::is_symlink()判断并使用fs::read_symlink()解析
总结

std::filesystem库的引入大幅简化了 C++ 文件系统操作,通过 RAII 设计和跨平台抽象,让开发者无需关心底层差异。

http://www.lryc.cn/news/576982.html

相关文章:

  • Spring生态创新应用
  • 学习React官方文档(描述UI)
  • 【机器学习实战笔记 13】集成学习:GBDT算法
  • (七)集成学习
  • stm32hal模块驱动(1)hpdl1414驱动
  • 机器学习7——神经网络上
  • 家谱族谱生成制作小工具PC/H5小程序开源
  • 【系统分析师】高分论文:论软件过程改进
  • UR Studio仿真工具上线助力协作机器人快速部署与精准配置模拟
  • Python 数据分析与可视化 Day 11 - 特征工程基础
  • 【GESP 四级】一个程序掌握大部分知识点
  • 【算法设计与分析】(三)二分搜索技术与大整数乘法
  • 信创背景下应用软件迁移解析:从政策解读到落地实践方案
  • vllm部署私有智谱大模型
  • AI算力综述和资料整理
  • Hive SQL 快速入门指南
  • 从理论到实战:解密大型语言模型的核心技术与应用指南
  • 理解 Confluent Schema Registry:Kafka 生态中的结构化数据守护者
  • 算法-基础算法-递归算法(Python)
  • 【C++11】异常
  • 【python】~实现工具软件:QQ邮件即时、定时发送
  • 预期功能安全SOTIF基本介绍
  • Kafka中的消费者偏移量是如何管理的?
  • 华为云Flexus+DeepSeek征文|基于华为云Flexus云服务快速搭建Dify-LLM应用开发平台详细教程
  • Springboot 集成 SpringState 状态机
  • Linux下的调试器-gdb(16)
  • Tcpdump 网络抓包工具使用
  • ali PaddleNLP docker
  • Vivado关联Vscode
  • BUCK电感电流检测电路current sense-20250603