最直接的方法是C++17起使用std::filesystem::exists(),返回bool表示路径是否存在(文件或目录),但不区分类型;需结合is_regular_file()判断是否为普通文件。
std::filesystem::exists() 最直接C++17 起,std::filesystem 是标准且跨平台的方案。调用 std::filesystem::exists(path) 即可返回 bool,表示路径是否存在(文件或目录都算存在)。
注意:它不区分文件和目录,只回答“这个路径有没有被系统识别”。如果需要确认是普通文件,得额外加 is_regular_file() 判断。
示例:
#includenamespace fs = std::filesystem; if (fs::exists("config.json")) { if (fs::is_regular_file("config.json")) { // 确实是文件,不是目录或符号链接 } }
-lstdc++fs(GCC 早期版本),Clang 可能需 -lc++fs;MSVC 默认支持std::string、const char* 或 fs::path,推荐统一用 fs::path 避免隐式转换问题false 误判GetFileAttributes() 更轻量不依赖 C++17,也不需链接额外库。调用 Windows API GetFileAttributesA() 或 GetFileAttributesW(),返回值为 INVALID_FILE_ATTRIBUTES 表示不存在。
示例(宽字符版,兼容中文路径):
#includeif (GetFileAttributesW(L"数据.csv") != INVALID_FILE_ATTRIBUTES) { // 存在(可能是文件、目录、甚至重解析点) }
INVALID_FILE_ATTRIBUTES 不代表一定是普通文件——目录、隐藏文件、系统文件都会通过FILE_ATTRIBUTE_DIRECTORY 位判断:& attr & FILE_ATTRIBUTE_DIRECTORY
access() 或 stat()
Linux/macOS 上常用 access(path, F_OK) 检查路径是否存在。它比 fopen() 开文件再关更轻量,也不

R_OK/W_OK)。
更严谨的做法是用 stat(),它能同时拿到类型、大小、修改时间等信息:
#includestruct stat buf; if (stat("log.txt", &buf) == 0) { if (S_ISREG(buf.st_mode)) { // 是普通文件 } }
access() 受进程有效 UID/GID 影响,可能因权限不足返回 -1,误判为“不存在”stat() 更可靠,但开销略高;若后续还需读取文件元信息,它是一次性收益setlocale(LC_CTYPE, "")
fopen() + fclose() 判断?常见误区:用 fopen("x.txt", "r") != nullptr 来判断文件存在。这看似可行,但有明显缺陷:
fclose()
fopen() 失败,但文件客观存在nullptr
除非你紧接着就要读内容,否则纯做存在性检查时,优先选 exists()、access() 或 stat() 这类轻量接口。
最易被忽略的一点:所有这些方法都只反映「调用瞬间」的状态。文件可能在你判断完、真正操作前就被删除或移动——需要原子性保证时,得靠 open() 的 O_CREAT | O_EXCL 等标志,而不是先查后开。