c - 使程序读取文件的某些部分

标签 c file sorting

我有一个包含学生姓名和成绩的文件,我想写一个程序,可以根据用户的选择对他们的成绩(如期中1、期中2)进行排序。我写的选择部分和打开的文件,但我不知道如何使程序只读文件的某些部分(例如,只有期中一年级)和排序他们只。这是我到目前为止写的东西;

#include <stdio.h>
#include <stdlib.h>

typedef struct {
int number;
char name[30];
char surname[30];
int midterm1,midterm2,midterm3;
} Student;

int main()
{
int choice,studentnumber,midterm1,midterm2,midterm3;
char surname;
FILE *cfPtr;

struct student *name;
name = malloc( 10 * sizeof(Student));

if ((cfPtr = fopen("grades.txt", "r")) == NULL)
printf("File cannot be opened.\n");
else {


const int STUDENTSMAX = 100;

Student students[STUDENTSMAX];
int i = 0;

while (!feof(cfPtr))

{
fscanf(cfPtr, "%d%s%s%d%d%d", &students[i].number,  &students[i].name,&students[i].surname, &students[i].midterm1, &students[i].midterm2, &students[i].midterm3);
 printf("%4d%15s%15s%10d%10d%10d\n", students[i].number, students[i].name,students[i].surname, students[i].midterm1, students[i].midterm2, students[i].midterm3);
 i++;
 }

 printf("What would you like to do? \n"
 "1- Sort according to midterm 1\n"
 "2- Sort according to midterm 2\n"
 "3- Sort according to midterm 3\n"
 "4- Exit\n");
  scanf("%d",&choice);

  while (choice != 4);{


  switch (choice) {

       case 1:
            qsort(students,10,sizeof(int),comp);
            for (i=0; i<9; i++)      
   printf("%4d%15s%15s%10d%10d%10d\n", students[i].number,  students[i].name,students[i].surname, students[i].midterm1);




    fclose(cfPtr);
   }

   system("PAUSE"); 
   return 0;
    }

最佳答案

考虑到可能是某种自由格式的文本文件(基于显示的代码),只读取整个文件(有点像您已经在做的)并只使用所需的部分可能是有意义的。如果文本文件具有固定偏移量的非常特定格式,则可以查找文件中的特定位置并读取特定列值,然后查找下一偏移量并从下一行读取列值。但这可能比它的价值更麻烦,也不会更有效率(如果有的话)。
话虽如此,要对结果进行排序,可能无论如何都需要整个文件。例如,如果您只是读取并排序“期中1”值,则结果将只是排序成绩,而不包含任何关联的姓名和学号。因此,在不了解更多目标的情况下,您可以考虑创建一个可以容纳一行的struct(学生编号、姓名、姓氏、midterm1等)。然后创建一个数组,并将每一行读入数组的一个元素中。如果你知道前面有多少行,你可以在一个块中分配数组,否则你可能需要重新分配它。
读取整个数组后,可以根据所需的值(例如,使用qsort)进行排序。
已经提到过,存在的问题/问题与现有的显示代码:
第二个printf的格式说明符%s少于参数。
第三个printf有“你想做什么”的问题,缺少结束部分。
fprintf不正确;它应该有一个文件句柄作为第一个参数。不过,我怀疑这可能是有意的?
最后一个printf循环在其结束paren后面有一个多余的分号(;),这意味着它有一个空的主体,而不是明显想要的whileprintf语句。
switch语句有点奇怪。我想那是“未完成”的部分。但把switch也包括进去似乎很奇怪。它可能在mainfclose的末尾。
使用else可能不是最佳选择。也许使用system("PAUSE");暂停输入更有意义。
编辑这里是一些额外的信息,以回应您的评论要求更多的细节。这听起来像是家庭作业,所以仅仅给出答案似乎是不对的。但有一种方法:
用文件中的6个项定义一个getch(基本上放在当前定义为局部变量的6个变量中)。
将局部变量(例如struct)声明为grades以构造您定义的结构。
使用pointer分配内存并将其分配给刚才提到的指针。记忆的数量也许是整个事情中最棘手的部分。malloc的size参数类似于malloc。问题是numRecs * sizeof( yourstruct )应该是什么。如果这是一个作业,你被告知会有多少记录(最大值),那么就用它。不过,如果这是“未知的”,那么有几种方法可以解决这个问题。一种是猜测一个数字(例如100),然后在循环中读取时,如果超过100,则使用numRecs。另一种替代方法(可能效率较低)是使用两个循环—一次读取它们而不存储它们,但只对它们进行计数,然后分配已知大小并再次读取它们。我会使用realloc版本。
将局部变量替换为数组(即realloced)。例如,您将使用malloc,而不是studentnumber。当你读的时候,记下有多少。然后可以使用grades[arraypos].studentnumber对数组进行排序。
编辑2
您的结构定义看起来是正确的,只是qsort成员不应该在其中。它仍然应该是一个局部变量。
为了便于使用,可以将结构定义为FILE *cfPtr;。这样您就可以在代码中使用typedef struct { ... } Student;而不是Student。注意,我将名称大写(命名时的个人偏好使其看起来不像变量名)。
对马洛克来说,你已经很接近了。但如前所述,它正在为单个记录分配空间。您需要这样的东西:struct Student(或name = malloc( 50 * sizeof( struct student ));,如果您将其更改为使用typedef。这假设不会有50条或更少的记录从文件中读取。

关于c - 使程序读取文件的某些部分,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11888642/

相关文章:

mysql - 稳定/可重复的随机排序(MySQL、Rails)

c - 为什么 "kill -15"有时会失败?

c# - .NET 文件锁定问题

java - 对 Arraylist 进行多次排序

python - python 中的合并排序 - 太慢?

java - Logic.class.getResource ("effects\\newball.wav");返回空值

c - 将指针分配给 const 函数参数指针

C 中的紧凑数组

c - 汇编中的 Malloc 和 strdup(自行实现)

java - 在java中解析XML的访问权限