c - 用 C 替换文本文件中的行

标签 c printf scanf fgets

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

文件结构为:

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.

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

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

关于c - 用 C 替换文本文件中的行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49131748/

相关文章:

c - 如何通过 C 中的函数初始化多个未知大小的数组

c - 如何在c中检查一行文本中的某些字符

c - C 中的 fprintf 无法正确打印

无法从c文件中读取字符串

c - 更新我的阵列棋盘游戏

c - 强制传输C中的所有串行数据

java - 在 Java 中打印大型列出数组

将 int 转换为 double

c - sscanf %u 不解释十六进制

C sscanf() 在读取 pgm 文件时返回零