使用fscanf()时不了解C格式说明符

时间:2014-12-12 03:19:10

标签: c c99 stdio format-specifiers cstdio

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

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都会将它放入自己的变量而不是连接它们。如果我使用点符号或直接 - >>它仍然有相同的结果。 谢谢:))

2 个答案:

答案 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