如何仅使用 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 争论,可能还需要一些宏帮助,但我认为它可以工作。