我有一个配置文件需要修改。
文件结构为:
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/