0

所以我正在阅读这种格式的文本文件:

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

如果我得到 print cp[0].lat; 我明白了。

 51.55531.555

当我打印 cp[0].lng; 我明白了。

 31.555

这是唯一正确的,我无法理解这种行为。为什么会这样?所有帖子都建议(正如我首先想到的那样) fscanf 中的每个 %s 都会将其放入自己的变量中,而不是将它们连接起来。如果我使用点表示法或直接表示法,则无关紧要 - >它仍然具有相同的结果。谢谢 :)

4

2 回答 2

1

格式说明符"%s[^... 尝试读取空格分隔的字符串,后跟字符[,然后是字符^。由于字符串将始终以空格结尾,因此下一个字符将始终是空格,它将不匹配[,并且格式说明符的其余部分都不匹配。

  • 始终检查返回值,fscanf以确保您阅读了您所做的所有事情。如果返回值错误,请给出诊断。

  • 读取固定大小的字符串数组时,始终使用字段大小限制。

所以在你的情况下,你想要的是:

if (fscanf(fp, "%3s%6s%6s", cp->name, cp->lat, cp->lng) != 3) {
    fprintf(stderr, "Incorrect data in input file, exiting!\n");
    abort(); }
于 2014-12-12T03:39:23.750 回答
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
于 2014-12-12T04:16:57.297 回答