c++如何读取配置文件的值_c++解析ini配置文件方法【实战】

3次阅读

INI文件读取本质是键值对解析,c++无标准库支持,常用inih库(单头文件、回调式解析)或手写解析器,windows下可用GetPrivateProfileString但需注意编码、路径和缓冲区问题。

c++如何读取配置文件的值_c++解析ini配置文件方法【实战】

INI 文件读取本质是键值对解析,C++ 没有标准库支持,必须借助第三方或手写解析器

标准 C++ 不提供 GetPrivateProfileString 或类似 INI 解析接口,Windows API 的 GetPrivateProfileString 仅限 Windows 平台且依赖 kernel32.lib,跨平台项目中基本不可用。多数实际项目选择轻量级第三方库(如 inih)或封装简单的行扫描逻辑——关键不是“能不能读”,而是“要不要支持节嵌套、注释、转义、空格容忍”等细节。

用 inih 库读取 INI 最省心,头文件即用,无需编译

inih 是纯 C 实现的单头文件库(ini.h + ini.c),C++ 项目可直接 #include 使用。它不构建内存结构,而是通过回调函数逐行通知解析结果,内存占用极低,适合嵌入式或配置项较少的场景。

典型用法:

// callback 函数 int handler(void* user, const char* section, const char* name, const char* value) {     if (strcmp(section, "database") == 0 && strcmp(name, "port") == 0) {         *(int*)user = atoi(value); // 假设 user 指向 int port;     }     return 1; // 继续解析 } 

// 调用 int port = 0; ini_parse("config.ini", handler, &port);

  • ini_parse 第二个参数是回调函数,第三个是传给它的上下文指针,适合传递多个变量地址(例如用 Struct 封装)
  • 不支持 [section:sub] 这类嵌套节名,遇到未知格式会跳过整行
  • 注释以 ;# 开头,会被自动忽略;键名前后空格被保留,值前后空格默认不 trim(需手动 strtrim

Windows 下用 GetPrivateProfileString 需注意路径和 Unicode 问题

若只面向 Windows 桌面程序,且配置文件路径固定,可用系统 API 快速读取,但容易踩坑:

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

  • GetPrivateProfileString 默认按 ANSI 编码读取,若 INI 文件是 UTF-8 无 bom,中文会乱码;建议改用 GetPrivateProfileStringW 并确保文件保存为 UTF-16 LE(记事本另存为时可选)
  • 路径必须是绝对路径或相对于当前工作目录;相对路径在 ide 调试时可能指向项目根目录,而发布后指向 exe 所在目录,建议用 GetModuleFileName 拼出配置文件全路径
  • 缓冲区大小必须足够,否则截断;返回值是实际复制的字符数(不含 ),不能直接当字符串长度用

示例:

TCHAR buffer[256] = {0}; GetPrivateProfileString(_T("network"), _T("host"), _T("localhost"),                          buffer, _countof(buffer), _T("app.ini")); // buffer 现在含 host 值,注意末尾有

手写简易解析器要防住换行、空行、注释和等号位置异常

如果项目不允许引入外部依赖,又只需读几个固定 key,手写解析器比想象中更可靠,但必须处理这些边界:

  • 一行含多个 =(如 url=http://a=b/c?d=e):应取第一个等号左边为 key,右边全部为 value
  • key 或 value 包含空格(如 log path = D:my logs):不能简单 strtok(" =", " t"),需手动找首个非空白后的 =
  • 注释在行尾(timeout=30 ; seconds):找到 = 后,再跳过右侧空格,检查是否以 ;# 开头
  • 空行、纯空格行、只有注释的行:跳过,不要当作 section 或 key

核心逻辑片段(C++17):

std::string line; while (std::getline(file, line)) {     auto pos = line.find_first_not_of(" trn");     if (pos == std::string::npos || line[pos] == ';' || line[pos] == '#') continue;     if (line[pos] == '[' && line.find(']', pos) != std::string::npos) {         // 解析 section     } else if (auto eq = line.find('='); eq != std::string::npos) {         std::string key = trim(line.substr(pos, eq - pos));         std::string val = trim(line.substr(eq + 1));         // 去掉 val 尾部的 ; 注释         if (auto cmt = val.find_first_of(";#"); cmt != std::string::npos) {             val.erase(cmt);         }         config[key] = trim(val);     } }

真正麻烦的从来不是“读到值”,而是“下次 INI 格式微调后,你的解析逻辑是否还健壮”。尤其当配置由运维手工修改时,多一个空格、少一个引号、错位的分号,都可能让程序静默读错值。宁可多花十分钟加几行 trim 和容错判断,也不要赌用户一定按文档格式写。

text=ZqhQzanResources