使用fscanf理解循环

Understanding a loop with fscanf

本文关键字:循环 fscanf 使用      更新时间:2023-10-16

我有一些代码,我想了解。我的c++知识有点生疏,所以也许我可以得到一些指针(双关语)来理解以下代码摘录:

vector<int> vArray;
FILE * fData;
fData = fopen ("some://path/file.txt", "r");
int iValue;
while(fscanf(fData, "%d", &iValue) != EOF){
 vArray.push_back(iValue);
}
fclose(fData);

我有以下两个问题:

  1. while环路与fscanf是如何准确工作的?它是按行读,按字符读,还是按其他方式读?
  2. 如果文件中有浮点值,如0.05、1.22或-5.3,会发生什么?有类型转换吗?

根据我的理解,vArray是一个只包含整数的向量,vArray.push_back(iValue)将把iValue的值作为一个新条目附加到向量上。我不明白的是给iValue赋值的方法。

fscanf函数返回成功匹配和赋值的输入项的数目,该数目可以小于规定的数目,或者在早期匹配失败的情况下甚至为零。

1。使用fscanf的while循环究竟是如何工作的?它是按行读,按字符读,还是按其他方式读?

fscanf从流中读取数据,并根据参数格式将数据存储到附加实参所指向的位置。

2。如果文件中有浮点值,比如0.05、1.22或-5.3,会发生什么?有类型转换吗?

如果你想读取浮点值,你应该使用%f。我认为你的循环只会读取整数。要了解更多信息,请查看此处的fscanf

scanf函数族,当以简单的方式(如问题代码)使用时,将首先读取并忽略任何空白,然后根据格式转换解析输入,直到下一个不兼容的字符进行该格式转换,通常是下一个空白。空格和换行符的处理方式相同,因此输入不需要以行为导向。

EOF常量被定义为-1,这是stdio函数在IO错误或文件结束时返回的值。因此,while循环将继续读取,直到fscanf遇到IO错误或文件结束。

然而,这段代码非常糟糕,因为scanf系列函数返回成功解析项的数量。如果文件中有非数字数据,它将无法解析代码中要求解析的1个整数。因此,它返回0而不是1,表示没有解析任何内容,并且它将把第一个无效字符留给流。

因此,如果文件fscanf中存在无效数据,则返回0 (iValue不变),而不是EOF,因此循环继续。接下来,fscanf再次读取无效数据,返回0,这就是无限循环。

顺便说一下,如果有包含.的浮点值,就会发生这种情况,因为在解析整数时,.是无效的char。

简单的修复方法是将condition更改为
while(fscanf(fData, "%d", &iValue) == 1){

则循环将在返回值EOF0时同时退出,并避免无限循环。在实际代码中,您可能希望以不同的方式处理文件结束符和解析错误。