c - C 中的 fscanf 动态数组?

标签 c arrays dynamic malloc scanf

所以,我的大作业是用 C 语言制作一个谁想成为百万富翁的游戏,但我有一个小问题。我有一个 csv 文件,其中包含 5000 行数据,如下所示:

...

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

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

15;...

...

我总是必须扫描 5000 个问题中的一个,总是需要比前一个难度更高(简单,我们有 15 个难度级别和 15 个问题),并且需要打印在屏幕上(所以我需要问题、答案和其他内容分开(不需要地质学))。但问题是:我必须始终使用 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);
}

最佳答案

Gergo,你让事情变得比他们需要的更加困难。首先,您肯定知道(或可以确定) proba.txt 中一行的最大长度是多少是。绝对不需要动态分配任何东西。只需声明onerow作为足够大的字符数组来容纳一行数据(包括 '\n'nul-termination 字符,并读取/丢弃 row - 1 行,然后对 fgets 进行最终调用阅读您感兴趣的行。

此外,请勿在代码中对文件名进行硬编码或使用其他魔数(Magic Number)。如果你需要常量,那么#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 行进入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 中的最后一个字符是'\n'验证是否读取了完整的行。确认最后一个字符是 '\n' 后char,我们只需用 nul-termination 字符覆盖来删除它,这样我们就没有 '\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 中就有您想要的行了。剩下的就是:

printf ("%s\n", onerow);

仔细检查一下,如果您还有任何其他问题,请告诉我。

关于c - C 中的 fscanf 动态数组?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47210686/

相关文章:

c - 对齐与打包属性

c - 在汇编中写一个 fopen 函数

C if 语句无法检测新行

c - 如何用assert实现C宏

c - 我想对c中的字符数组进行排序

javascript - 在react js中独立更改<td>的颜色

arrays - 不能使用 "a"(字符串类型)作为 go 中数组元素的类型

c# - C# 中使用动态 SQL 的变量表名

Java浏览器,动态字符串匹配器模式

php - 使用 PHP 的动态表