所以我正在阅读这种格式的文本文件:
ABC 51.555 31.555
DEF 23.445 45.345
我正在尝试使用fscanf()来解析数据,因为这个文件可以增长或缩小它需要在加载方式时是动态的,因此我使用malloc并且我也想将它存储在下面的结构中。我认为问题在于空间,甚至可能没有正确编写整个格式说明符。这是我的代码。
typedef struct data
{
char name[4];
char lat[7];
char lng[7];
}coords;
int main(int argc, char *argv[])
{
////////////CREATES FILE POINTER/////////
FILE* fp;
///////////CREATES MALLOC POINTER TO STORE STRUCTS/////////////
coords* cp;
//////////OPENS FILE//////////
fp = fopen(argv[1], "r");
/////////GET THE TOTAL AMMOUNT OF LINES IN THE FILE/////////
fseek(fp, 0, SEEK_END);
long size = ftell(fp);
rewind(fp);
//////SKIPS FIRST LINE//////////
while(fgetc(fp) != (int)'\n')
{};
/////////ASSIGNS MEMORY THE SIZE OF THE FILE TO //////////
cp = malloc(sizeof(coords) * size);
//////////READS FILE AND STORES DATA///////
fscanf(fp,"%s[^ ] %s[^ ] %s[^\n]", cp->name, cp->lat, cp->lng);
printf("%s\n%lf\n%lf\n", cp->name, cp->lat, cp->lng);
fclose(fp);
return 0;
}
是的,我知道我没有包含头文件,但我有正确的stdlib和stdio
更新1: 我已经尝试了两个回复,我在屏幕上看到了这个:
ABC51.555
0.000000
0.000000
为什么51.555没有进入结构中的下一个项目? 感谢
/////////////////////////////////////////////// ////////////////更新2 //////////////////////////////// ////////////////////////
好的,我已修改我的代码以执行以下操作。
typedef struct data
{
char name[4];
char lat[6];
char lng[6];
}coords;
int main(int argc, char *argv[])
{
////////////CREATES FILE POINTER/////////
FILE* fp;
///////////CREATES MALLOC POINTER TO STORE STRUCTS/////////////
coords* cp;
//////////OPENS FILE//////////
fp = fopen(argv[1], "r");
/////////GET THE TOTAL SIZE OF THE FILE/////////
fseek(fp, 0, SEEK_END);
long size = ftell(fp);
long lines = -1;
rewind(fp);
//////GETS TOTAL AMMOUNT OF LINES/////////
char c;
while(c != EOF)
{
c = fgetc(fp);
if(c == '\n')
{
lines++;
}
}
rewind(fp);
////////////SKIPS FIRST LINE//////////
while(fgetc(fp) != (int)'\n')
{};
/////////ASSIGNS MEMORY THE SIZE OF THE FILE TO //////////
cp = malloc(sizeof(coords) * size);
//////////READS FILE AND STORES DATA///////
printf("Lines of text read: %d\n", lines);
fscanf(fp,"%s %s %s[^\n]", cp[0].name, cp[0].lat, cp[0].lng);
printf("%s\n", cp[0].name);
fclose(fp);
return 0;
}
现在当我尝试打印cp [0] .name;我得到了整个第一行没有空格,就像这样。
ABC51.55531.555
如果我得到了打印cp [0] .lat;我明白了。
51.55531.555
当我打印cp [0] .lng;我明白了。
31.555
哪个是唯一正确的,我无法理解这种行为。它为什么会这样?所有帖子都建议(正如我首先想到的)fscanf中的每个%s都会将它放入自己的变量而不是连接它们。如果我使用点符号或直接 - >>它仍然有相同的结果。 谢谢:))
答案 0 :(得分:1)
格式说明符"%s[^
...尝试读取whitspace分隔的字符串,后跟字符[
,然后是字符^
。由于字符串将始终以空格结尾,因此下一个字符将始终为空格,不会与[
匹配,并且格式说明符的其余部分都不会匹配。
总是检查fscanf
的返回值,以确保您阅读了所做的所有事情。如果返回值错误,请进行诊断。
在读入固定大小的字符串数组时,始终使用字段大小限制。
所以在你的情况下你想要的是:
if (fscanf(fp, "%3s%6s%6s", cp->name, cp->lat, cp->lng) != 3) {
fprintf(stderr, "Incorrect data in input file, exiting!\n");
abort(); }
答案 1 :(得分:0)
我不确定您是否要使用空格分隔符[^]。 fscanf已经将空白字符串解析为默认值。试试这个,看看是否正确解析了字符串:
fscanf(fp, "%s %s %s[^\n]", cp->name, cp->lat, cp-lng);
输出应该导致:
cp->name ---- ABC
cp->lat ----- 51.555
cp->lng ----- 31.555