c - Clang-Tidy: 'fscanf' 用于将字符串转换为整数值,但函数不会报告转换错误;考虑改用“strtol”
问题描述
对这里的 C 有点陌生!
我正在使用 CLion 编写程序,并且每当我fscanf
将输入文件中的值存储到变量中时,都会不断收到此警告消息:
Clang-Tidy: 'fscanf' 用于将字符串转换为整数值,但函数不会报告转换错误;考虑改用“strtol”
我不明白这个错误,因为我认为fscanf
是我应该用来读取输入文件的函数?有人可以解释(在菜鸟级别)我使用它的方式有什么问题吗?
这是我的代码示例:
FILE *initial_configuration_box_1_pointer; initial_configuration_box_1_pointer = fopen("initial_configuration_box_1.xyz", "r"); fscanf(initial_configuration_box_1_pointer, "%d\n", &N1); // Warning here. fscanf(initial_configuration_box_1_pointer, "pid\tx\ty\tz\n"); // No warning here. for (i = 1; i <= n1; i++) { fscanf(initial_configuration_box_1_pointer, "p\t%lf\t%lf\t%lf\n", &rx1[i - 1], &ry1[i - 1], &rz1[i - 1]); // Warning here. } fclose(initial_configuration_box_1_pointer);
我知道有人问过类似的问题,但我无法理解他们得到的任何(少数)答案......
解决方案
初学者完全避免 scanf 有很多充分的理由。(阅读http://sekrit.de/webdocs/c/beginners-guide-away-from-scanf.html)。如果您要以交互方式使用它,切勿以空格结尾格式字符串,因为这样做只会让用户感到困惑。并始终检查 scanf 返回的值以查看它是否实际匹配任何输入。一个常见的错误是输入无法匹配预期的数据,并且 scanf 循环变成了一个无限循环,重复检查相同的无效输入。也许警告暗示了上面的第二点:由于 scanf 不会为您验证输入,因此您必须通过检查有多少转换说明符scanf
能够匹配来明确地进行验证。尝试类似:
#include <stdio.h>
#include <stdlib.h>
int
main(int argc, char ** argv)
{
int n1;
const char *path = argc > 1 ? argv[1] : "initial_configuration_box_1.xyz";
FILE *ifp;
if( (ifp = fopen(path, "r")) == NULL ){
perror(path);
return EXIT_FAILURE;
}
if( 1 != fscanf(ifp, "%d", &n1) || n1 <= 0 ){
fprintf(stderr, "Invalid input\n");
return EXIT_FAILURE;
}
fscanf(ifp, " pid x y z");
double rx1[n1];
double ry1[n1];
double rz1[n1];
for( int i = 0; i < n1; i++ ){
if( 3 != fscanf(ifp, " p %lf %lf %lf", rx1 + i, ry1 + i, rz1 + i) ){
fprintf(stderr, "Invalid input near line %d\n", i);
return EXIT_FAILURE;
}
}
if( fclose(ifp) ){
perror(path);
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
请注意,scanf
格式字符串中的空格不完全匹配,因此使用\n
or\t
或
完全一样。通常,人们只使用一个空格来使其更易于阅读。此外,某些转换说明符(值得注意的%d
and %lf
)之间的空格是无关紧要的,只是为了可读性而包含在内。