超级好用的配置文件解析器:minIni
哈喽,我是老吴。这两天发现一个还不错的开源项目,记录一下学习心得。
对于嵌入式开发,INI 文件使用门槛低、用途广,掌握一个稳定的开源 INI 解析器可以提升编码效率、避免重复造轮子。
另外,对于一个 C 初学者,MinIni 代码严谨且健壮,API设计合理,是一个非常适合用来练习文本读写、字符串操作的开源项目。
minIni 是什么?
https://github.com/compuphase/minIni
minIni 是一个用于读取和写入 INI 文件的库。
什么是 INI 文件?
INI 是一种有简单语法的纯文本文件,例如:
[Network]
hostname=My Computer
address=dhcp
dns=192.168.1.1
timeout=10
[Network] 是一个 Section,Section 下面有多个键值对。
INI 文件的特点是简单易用、可读性好。
Linux 系统 etc 目录下的许多配置文件都是 INI 文件,只是有的配置文件没有显式声明 Section。
例如 bluez 的配置文件 /etc/bluetooth/main.conf:
[General]
# Default adaper name
Name = BlueZ
[Policy]
# AutoEnable defines option to enable all controllers when they are found.
AutoEnable=true
minIni 的几个特点:
1> minIni 大约是 950 行代码 (包括注释),是一个真正的 “迷你” INI 文件解析器,非常容易移植到各种嵌入式平台。
2> minIni 不需要标准 C/C++ 库中的 文件 I/O 函数,且允许通过宏配置要选择文件 I/O 接口。
3> minIni 仅使用 stack ,完全不使用动态内存(malloc)。
4> 有 C++ binding, 能很好地配合 C++ 使用。
minIni 怎么用?
minIni 的核心源码就一个 .c 和 .h 文件,我们直接将它们集成到项目代码里即可。
快速地了解它的 API:
#include "minIni.h"
#define sizearray(a) (sizeof(a) / sizeof((a)[0]))
const char inifile[] = "example.ini";
int main(void)
{
char str[100];
char section[50];
long n;
n = ini_gets("Network", "address", "dummy", str, sizearray(str), inifile);
if (n >= 0) printf("Network/address=%s", str);
n = ini_getl("Network", "timeout", -1, inifile);
printf("Network/timeout=%ld\n", n);
待解析的文件内容:
[Network]
hostname=My Computer
address=dhcp
dns=192.168.1.1
timeout=10
运行结果:
Network/address=dhcp
Network/timeout=10
ini_gets() 用于获取字符串类型的值。
参数 1 是 Section name;
参数 2 是 Key name;
参数 3 是获取不到值时的默认值;
参数 4 是用于保存目标键值的 Buffer;
参数 5 是 Buffer 的长度;
参数 6 是 INI 文件的路径;
ini_getl() 用于获取整型类型的值,参数的含义和 ini_gets() 大同小异。
完整的 API 列表:
minIni 的内部实现?
我简单地浏览过 minIni 的实现代码,感觉良好。
想实现解析 INI 文件的功能并不难,难的是要认真考虑各种边界情况和异常情况,保证程序的健壮性和 API 的合理性。
以 ini_gets() 为例:
就是先打开文件,然后用 getkeystring() 找到目标键值,最后拷贝给调用者。
getkeystring() 负责真正地解析文本:
大约有 80 行代码,我简单地总结一下思路:
1> 用 fgets 进行逐行读取,用 strrchr 找到包含 '[' 和 ']' 的行,然后再用 strncasecmp 找到目标 Secntion 所在的行。
2> 继续用 fgets 进行逐行读取,用 strrchr 找到包含 '=' 的行,然后再用 strncasecmp 找到目标 Key 所在的行。
3> 用 strncpy 将目标 Key 的值拷贝给调用者。
虽然解析文本只需要做到这 3 个步骤,但是由于需要对各种边界情况进行判断,以及处理空格之类的无效字符,所以需要细心且反复调试才能编写好这个函数。
总结
对于嵌入式开发,INI 文件使用门槛低、用途广,掌握一个稳定的开源 INI 解析器可以提升编码效率、避免重复造轮子。
对于一个 C 初学者,MinIni 代码严谨且健壮,API设计合理,是一个非常适合用来练习文本读写、字符串操作的开源项目,值得一看。
其他 INI 解析的开源项目:
https://github.com/rxi/ini
https://github.com/benhoyt/inih