用 C 替换文本文件中的行

Replacing lines in a text file with C

我有一个配置文件需要更改。

文件结构为:

resolution 12x34
interval 1234

所以两个带有空格分隔符的字符串。我用来改变它的代码是这样的:

FILE *fp = fopen(configuration_file, "a+");
char str[100], key[100], value[100];
if(fp) {
    while(fgets(str, 100, fp) != NULL) {
        if(2 == sscanf(str, "%s %s", &key, &value)) {
            if(strcmp(key, "resolution") == 0){
                if(msg->resolution){
                    fprintf(fp, "%s %s\r\n", key, msg->resolution);
                }
            } else if(strcmp(key, "interval") == 0) {
                if(msg->interval) {
                    fprintf(fp, "%s %d\r\n", key, msg->interval);
                }
            } else {
                fputs(str, fp);
            } 
        } else {
            fputs(str, fp);
        }
    }
} else {
    (void)printf("-- Configuration file not found (%s)\r\n --", configuration_file);
}
fclose(fp);

我的想法是逐行阅读。根据 fgets 的文档,它在换行符处停止。字符串扫描每一行并将它们解析为键和值。到目前为止一切顺利,按预期行事。然后将新行打印到文件中,覆盖它刚刚扫描的行。这就是问题所在。如果我使用 fprintf,则只处理第一个值 resolution。结果是:

resolution oldxres
resolution newxres

它会覆盖错误的行并完全跳过第二行。

如果我删除 fprintf 而不是简单地打印它找到的值,它会按预期打印这两个值。

我在这里错过了什么? fprintf 是否推动文件指针?

And then print the new line to the file, overwriting the line it had just scanned.

文件不能这样工作。写入新文件。完成后,将新文件重命名为旧名称。或者,将整个文件读入内存,更改内存中的内容,然后将其写回。

如果修改后的行永远不会比原始行长,您可以重写个别行。否则,修改后的行将溢出您尚未阅读的下一行,并将其销毁。为了防止这种情况,您需要某种 look-ahead 缓冲区,这太麻烦而且 error-prone。在最坏的情况下,您无论如何都需要读取整个文件。