C中的fscanf动态数组?

时间:2017-11-09 20:09:40

标签: c arrays dynamic malloc scanf

所以,我的大作业是让一个想要在C中成为百万游戏的人,但我有一点问题。我有一个包含5000行数据的csv文件,如下所示:

...

4;我多大了?; 22;狗;黄色;谁知道; A;个人

2;乌得勒支在哪里?;英格兰;尼瑟兰;法国;瑞典; B;地质学

15; ...

...

我总是要扫描一个5000的问题,总是需要比以前更难(简单,我们有15个级别的困难和15个问题),并且需要在屏幕上打印(所以我需要问题,答案和其他人分开(GEOLOGY不需要))。但问题是:我必须始终使用malloc!怎么可能知道字符串会有多长?

我在这里:

    srand(time(NULL));
    int num=1;
    char* onerow=(char*) malloc(num*sizeof(char));
    int row=rand()%1000;
    char c;

    FILE *fp;
    fp = fopen("proba.txt", "r");

    for (int i=0; i<=row-1; i++) {
        if(i!=row-1) {          //not the row we need, jump over it
            while(c!='\n')
                c=fgetc(fp);
        }
        if(i==row-1) {
            for(int j=0; onerow[j]!='\n'; j++) {    //this is what we need
                onerow[j]=fgetc(fp);
                onerow = (char *) realloc(onerow, ++num*sizeof(char));
            }
        }
    }
    onerow[num]='\0';
    fclose(fp);
    printf("%s\n", onerow);
    free(onerow);
}

1 个答案:

答案 0 :(得分:2)

Gergo,你正在使自己的事情变得比他们需要的更难。首先,你肯定知道(或可以确定)proba.txt中一行的最大长度是多少。绝对没有必要动态分配任何东西。只需将onerow声明为足够大的字符数组,以容纳一行数据(包括'\n' nul-terminatedating 字符,并读取/放弃row - 1行然后最后拨打fgets以阅读您感兴趣的行。

此外,请勿在代码中硬编码文件名或使用其他幻数。如果您需要常量,那么#define将它们放在最上面,这样您就可以根据需要进行更改。例如,要设置最大行长度和文件名,您可以使用以下内容:

#define MAXR 512            /* max chars per line (adjust as required */
#define FNAME "proba.txt"   /* filename */
...

接下来,所有数组和C中的计数都基于zero。虽然您可以根据自己喜欢的任何数字进行统计,但保持一致可以帮助您牢记事情。因此,num0开始,因为row = rand() % 1000;返回0-999的值。进行这些更改后,您的声明可能如下所示:

srand(time(NULL));          /* seed the random number generator */

int num = 0,                /* counter variable */
    row = rand() % 1000;    /* row to read */
size_t len = 0;             /* length of row */
char onerow[MAXR] = "", c;
FILE *fp = fopen (FNAME, "r");

注意:您必须验证您的文件实际上已打开以供阅读,否则您将调用未定义行为尝试从非读取 - 打开文件流,例如\

if (!fp) {  /* validate file open for reading */
    fprintf (stderr, "error: file open failed '%s'\n", FNAME);
    return 1
}

如上所述,一旦文件打开,您需要做的就是阅读并丢弃row-1行,然后阅读最后一次阅读row th 排成onerow然后关闭文件,例如

while (num++ < row)             /* read/discard row - 1 lines */
    fgets (onerow, MAXR, fp);
fgets (onerow, MAXR, fp);       /* read row line */
fclose (fp);                    /* close file */

由于我们使用fgets(一个面向行的输入函数),我们必须验证 onerow中的最后一个字符是{{ 1}}验证是否已读取完整的行。在确认最后一个char是'\n' char后,我们只需通过使用 nul-terminatedating 字符覆盖它来删除它,这样我们就没有'\n'悬挂在...字符串。只需获取长度,检查并覆盖:

'\n'

如果你已经做到这一点,你可以在len = strlen (onerow); /* get onerow length */ if (onerow[len - 1] == '\n') /* validate last char is '\n' */ onerow[--len] = 0; /* overwrite with nul-character */ else { fprintf (stderr, "error: line '%d' exceeds '%d' chars.\n", num, MAXR); return 1; } 中找到所需的行,剩下的就是:

onerow

仔细看看,如果您有任何其他问题,请告诉我。