如何仅使用 fread() 和 vsscanf() 包装 fscanf()
How to wrap fscanf() using only fread() and vsscanf()
我正在使用类 C API 的嵌入式平台上移植一些代码。原始代码使用 fscanf()
从文件中读取和解析数据。不幸的是,在我的 API 上,我没有 fscanf()
等效项,因此在实际移植之前,我试图使用 fread()
和 vsscanf()
(我有)。我也有相当于 fseek()
和 ftell()
.
编辑:请记住,对嵌入式文件系统的访问非常有限(fread - fseek - ftell - fgetc - fgets),所以我需要一个有效的解决方案在内存中使用字符串,而不是以其他方式访问文件。
代码看起来像这样:
int main()
{
[...] /* variable declarations and definitions */
do
{
read = wrapped_fscanf(pFile, "%d %s", &val, str);
} while (read == 2);
fclose(pFile);
return 0;
}
int wrapped_fscanf(FILE *f, const char *template, ...)
{
va_list args;
va_start(args, template);
char tmpstr[50];
fread(tmpstr, sizeof(char), sizeof(tmpstr), f);
int ret = vsscanf(tmpstr, template, args);
long offset = /* ??? */
fseek(f, offset, SEEK_CUR);
va_end(args);
return ret;
}
问题是 fscanf()
将指针移动到匹配结束时文件流中的位置,而 fread()
我正在读取固定数量的数据(在此case 50 bytes),我应该找到一种方法将指针移回匹配字符串的末尾。
假设我从文件中读取的 50 个字符的字符串如下:
12 bar 13 foo 56789012345678901234567890123456789
fscanf()
将匹配 int 12
,字符串 bar
并且指针将指向 "bar" 中的 "r" 之后,因此我可以调用再次阅读 13 foo
另一方面,fread()
将指针放在 50 元素序列中的最后一个字符之后,这是错误的:我仍然必须阅读 13 foo
但如果我调用 wrapped_fscanf()
指针再次位于第 51 个位置。
我必须使用 fseek()
回滚到第一场比赛的结尾,但我该怎么做?如何计算 offset
的值?
vsscanf()
returns 匹配的数量,而不是字符串的长度,我无法知道有多少空白字符分隔匹配的元素( 或我呢?)
即我得到与
相同的输出( {var,str,read} == {9,"xyz",2} )
9 xyz
和
9 xyz
是否有一些我不知道的技巧,或者除了用 fread()
vsscanf()
ftell()
和 [ 包装 fscanf()
之外,我是否必须找到另一种解决方案=18=]?
谢谢
您将需要一些中间缓冲代码,这些代码将抓取数据块(使用 fread),并扫描您的缓冲区以查找模式。如果找到模式,则截断缓冲区,如果未找到模式,则追加更多数据。这实际上是 fscanf 将要做的事情。
假设您的 vsscanf()
实现支持它,您对 fscanf()
的替代可以将 %n
字段描述符附加到所提供格式的末尾。只要在 vsscanf()
到达该字段之前没有失败,它就会将到该点消耗的字符数存储在相应的参数中。然后您可以使用该结果适当地重新定位流。这将需要一些 varargs 争论,可能还需要一些宏帮助,但我认为它可以工作。
我正在使用类 C API 的嵌入式平台上移植一些代码。原始代码使用 fscanf()
从文件中读取和解析数据。不幸的是,在我的 API 上,我没有 fscanf()
等效项,因此在实际移植之前,我试图使用 fread()
和 vsscanf()
(我有)。我也有相当于 fseek()
和 ftell()
.
编辑:请记住,对嵌入式文件系统的访问非常有限(fread - fseek - ftell - fgetc - fgets),所以我需要一个有效的解决方案在内存中使用字符串,而不是以其他方式访问文件。
代码看起来像这样:
int main()
{
[...] /* variable declarations and definitions */
do
{
read = wrapped_fscanf(pFile, "%d %s", &val, str);
} while (read == 2);
fclose(pFile);
return 0;
}
int wrapped_fscanf(FILE *f, const char *template, ...)
{
va_list args;
va_start(args, template);
char tmpstr[50];
fread(tmpstr, sizeof(char), sizeof(tmpstr), f);
int ret = vsscanf(tmpstr, template, args);
long offset = /* ??? */
fseek(f, offset, SEEK_CUR);
va_end(args);
return ret;
}
问题是 fscanf()
将指针移动到匹配结束时文件流中的位置,而 fread()
我正在读取固定数量的数据(在此case 50 bytes),我应该找到一种方法将指针移回匹配字符串的末尾。
假设我从文件中读取的 50 个字符的字符串如下:
12 bar 13 foo 56789012345678901234567890123456789
fscanf()
将匹配 int 12
,字符串 bar
并且指针将指向 "bar" 中的 "r" 之后,因此我可以调用再次阅读 13 foo
另一方面,fread()
将指针放在 50 元素序列中的最后一个字符之后,这是错误的:我仍然必须阅读 13 foo
但如果我调用 wrapped_fscanf()
指针再次位于第 51 个位置。
我必须使用 fseek()
回滚到第一场比赛的结尾,但我该怎么做?如何计算 offset
的值?
vsscanf()
returns 匹配的数量,而不是字符串的长度,我无法知道有多少空白字符分隔匹配的元素( 或我呢?)
即我得到与
相同的输出( {var,str,read} == {9,"xyz",2} )9 xyz
和
9 xyz
是否有一些我不知道的技巧,或者除了用 fread()
vsscanf()
ftell()
和 [ 包装 fscanf()
之外,我是否必须找到另一种解决方案=18=]?
谢谢
您将需要一些中间缓冲代码,这些代码将抓取数据块(使用 fread),并扫描您的缓冲区以查找模式。如果找到模式,则截断缓冲区,如果未找到模式,则追加更多数据。这实际上是 fscanf 将要做的事情。
假设您的 vsscanf()
实现支持它,您对 fscanf()
的替代可以将 %n
字段描述符附加到所提供格式的末尾。只要在 vsscanf()
到达该字段之前没有失败,它就会将到该点消耗的字符数存储在相应的参数中。然后您可以使用该结果适当地重新定位流。这将需要一些 varargs 争论,可能还需要一些宏帮助,但我认为它可以工作。