2

我正在使用 fscanf 读取日期,然后使用 fgets 读取注释。然而,在第一次迭代之后, fscanf 返回值 -1。

我用GDB一步步调试程序。在第一次使用 fgets 之前它工作正常。当我尝试打印 fgets 在第一次迭代中读取的行时,它给了我这个:

(gdb) print line
$6 = "\rtest\r18/04/2010\rtest2\r03/05/2010\rtest3\r05/08/2009\rtest4\r\n\000\000\000\000q\352\261\a\370\366\377\267.N=\366\000\000\000\000\003\000\000\000\370xC\000\000\000\000\000\000\000\000\000\001\000\000\000\227\b\000\000\070\367\377\267H\364\377\267\362\202\004\bdoD\000\354\201\004\b\001\000\000\000\304oC\000p\363\377\277\260zC\000D\363\377\277\n!B\000\064\363\377\277\354\201\004\b(\363\377\277TzC\000\000\000\000\000\070\367\377\267\001\000\000\000\000\000\000\000\001\000\000\000\370xC\000\001\000\000\000\000\000\312\000\000\000\000\000\377\260\360\000\001\000\000\000\277\000\000\000\364\317\000\000\344\261\\\000\000\000\000\000p\363\377\277|\233\004\b\350\362\377\277 \204\004\b\005\000\000\000|\233\004\b\030\363\377\277"

看起来 fgets 读取了剩余的条目,然后将它们全部存储在一个字符串中。

我不确定它为什么这样做。

这是主要代码:

int main(int argc, char* argv[]) {
    FILE* file;
    int numEntries, i = 0;
    int index = atoi(argv[1]);
    char line[SIZE];
    JournalEntry *entry;

    /*argument provided is the entry user wants to be displayed*/
    if (argc > 2) {
        perror("Error: Too many arguments provided");
    }
    file = fopen("journalentries.txt", "r");
    if (file == NULL) {
        perror("Error in opening file");
    }

    if (fscanf(file, "%d", &numEntries) != 1) {
        perror("Unable to read number of entries");
    }

    entry = (JournalEntry*)malloc(numEntries  * sizeof(JournalEntry));
    if (entry == NULL) {
        perror("Malloc failed");
    }

    for (i = 0; i < numEntries; i++) {
        if (fscanf(file, "%d/%d/%d", &entry[i].day, &entry[i].month, &entry[i].year) != 3) {
            perror("Unable to read date of entry");
        }

        if (fgets(line, sizeof(line), file) == NULL) {
            perror("Unable to read text of entry");
        }
    }

    printf("%d-%02d-%02d %s: ", entry[index].year, entry[index].month, entry[index].day, entry[index].text);

    if(ferror(file)) {
        perror("Error with file");
    }

    fclose(file);
    free(entry);

    return 0;
}

我必须阅读的文件:第一行包含要读取的条目数

4
12/04/2010
test
18/04/2010
test2
03/05/2010
test3
05/08/2009
test4

位于头文件中的结构 JournalEntry:

typedef struct {
    int day;
    int month;
    int year;
    char text[250];
} JournalEntry;
4

3 回答 3

1

看起来 fgets 读取了剩余的条目,然后将它们全部存储在一个字符串中。

是的,'\r'不是行终止符。因此,当fscanf在第一个无效字符处停止解析并将它们留在缓冲区中时,fgets将读取它们直到行尾。并且由于文件中没有有效的行终止符,直到文件结束。

您可能应该将文件修复为具有有效的(Unix?)行结尾,例如使用合适的文本编辑器可以做到这一点。但这是另一个问题,之前已经提出过(如此),并且取决于您的问题中未包含的详细信息。

此外,您需要双重检查fscanf返回值。仅在返回值为 -1 时使用perror,否则错误消息将与错误完全无关。如果返回值>=0与您想要的不同,则打印自定义错误消息“输入语法无效”或其他任何内容(并可能用于fgets从缓冲区中读取其余行)。

此外,为了可靠地混合scanfand fgets,我需要在fscanf格式字符串中添加空格,因此它将读取行尾的任何空格(也在下一行的开头和任何空行,所以如果这很重要,请小心), 像这样:

int items_read = scanf("%d ", &intvalue);

正如另一个答案中所述,最好只读取行fgets,然后sscanf逐行解析它们。

于 2013-10-14T15:47:19.740 回答
1

不要混合fscanf()and fgets(),因为前者可能会在流的缓冲区中留下一些东西。

对于面向行的格式,使用 仅读取整行fgets(),然后使用 egsscanf()来解析您所阅读的内容。

于 2013-10-14T15:47:59.777 回答
0

您在运行 GDB 时看到的字符串实际上以第一个空字符结束:

"\rtest\r18/04/2010\rtest2\r03/05/2010\rtest3\r05/08/2009\rtest4\r\n\000"

后面的其他数据被忽略(使用普通 str 函数时);

于 2013-10-14T15:50:28.070 回答