c中的文本文件(int,string,string和float)的fscanf和fgets

时间:2017-12-03 12:02:05

标签: c string file scanf fgets

我正在尝试使用文本文件创建4个数组。文本文件如下所示:

1000 Docteur             Albert              65.5
1001 Solo                Hanz                23.4
1002 Caillou             Frederic            78.7
…

代码:

void creer (int num[], char pre[][TAILLE_NP+1], char nom[][TAILLE_NP+1], 
float note[], int * nb ){

  int  n = 0, i; /*nb personnes*/

  FILE *donnees = fopen("notes.txt", "r");

  if(!donnees){
    printf("Erreur ouverture de fichier\n");
    exit(0);
  }

  while (!feof(donnees)){

    fscanf(donnees,"%d", &num [n]);
    fgets(nom[n], TAILLE_NP+1, donnees);
    fgets(pre[n], TAILLE_NP+1, donnees);
    fscanf(donnees,"%f\n", &note[n]);

    printf("%d %s %s %f\n",num[n], nom[n], pre[n], note[n]);
    n++;
    }

  fclose (donnees);

  *nb = n ;
  }


int main() {

  int num[MAX_NUM];
  int nbEle;

  char pre[MAX_NUM][TAILLE_NP+1],
       nom[MAX_NUM][TAILLE_NP+1];

  float note[MAX_NUM];

  creer (num, pre, nom, note, &nbEle);

  printf("%s", pre[2]); //test

  return 0; 
}

问题是,我确信有一种更好的方法来创建数组,我是初学者。另外,浮点数有问题,当我打印时,小数点不正确。例如,78.7变为78.699997。 我究竟做错了什么? 谢谢 ! :)

2 个答案:

答案 0 :(得分:1)

这里有两个问题:

  1. 混合fscanf()fgets()是一个坏主意,因为前者适用于行的某些部分,后者适用于完整的行。

  2. float并不像您预期​​的那样精确。

  3. 解决1:

    fscanf(donnees, "%d", &num[n]);
    fscanf(donnees, "%s", nom[n]);
    fscanf(donnees, "%s", pre[n]);
    fscanf(donnees, "%f\n", &note[n]);
    

    为了避免溢出“字符串”,您可以告诉fscanf()以最大值扫描多少,例如"%42s"用于42 char s的字符串(不是计算0 - 终止char)。

    要解决2:

    note设为double并执行

    fscanf(donnees,"%lf\n", &note[n]);
    

答案 1 :(得分:0)

这里有几个问题:

浮点非常棘手。阅读floating-point-gui.de(并记住该网址)。

不要在automatic variables上分配巨额call stack。典型的调用帧应该不超过几千字节(并且您的整个调用堆栈应该小于一个或几兆字节)。使用C dynamic memory allocation

C只有一维数组。如果您需要更好,请创建一些抽象数据类型(通常,避免数组数组)。看看this的灵感。

仔细阅读每个标准功能的documentation。您应该测试fscanf的结果。