从文件读数据插入到链表
前两周做了一个小作业,学生成绩管理系统,第一周实现了录入学生信息、删除学生信息、显示学生信息和按照学生平均成绩排序的功能,总体来说比较顺利,第二周只做了一件事就是读txt文件中的学生信息,将txt文件中的学生信息读到程序中插入到链表中,这一个看似简单的工作,花费了一周的时间。
我程序中用到的txt文件中的数据是学生成绩,其中有学生ID、学生姓名、成绩等,在链表中分别是int型,字符串和float型,如下所示。
因为之前学习了I/O文件的操作,没有学习基于流的I/O,所以一开始就是瞎撞,根本不知道怎么解决,把这些信息读出来好像也没有什么用,想的最多的就是程序怎么把这些文件中的信息分别读出来,ID是ID、成绩是成绩、姓名是姓名,这就是一开始的思路。然后开始看书,上网搜索都没有直接的信息,看了其他的程序都是用基于流的I/O操作,用fopen等函数来操作文件的。先学习了fopen函数和fread函数。
fopen(const char *path,cost char *mode)
头文件: #include<stdio.h>
参数说明:
第一个参数*path表示要打开文件的路径;
第二个参数*mode代表打开的方式,mode有以下几种值:
r:只读方式打开,文件必须存在
r+:可读写,必须存在
rb+:打开二进制文件,可以读写
rt+:打开文本文件,可读写
w:只写,文件存在则文件长度清0,文件不存在则建立该文件
w+:可读写,文件存在则文件长度清0,文件不存在则建立该文件
a:附加方式打开只写,不存在建立该文件,存在写入的数据加到文件尾,EOF符保留
a+:附加方式打开可读写,不存在建立该文件,存在写入的数据加到文件尾,EOF符不保留
wb:打开二进制文件,只写
wb+:打开或建立二进制文件,可读写
wt+:打开或建立文本文件,可读写
at+:打开文本文件,可读写,写的数据加在文本末尾
ab+:打开二进制文件,可读写,写的数据加在文件末尾
由mode字符可知,上述如r、w、a在其后都可以加一个b,表示以二进制形式打开文件。
返回值:文件打开成功返回一个指向该打开文件的指针(FILE结构);文件打开失败,错误上存error code。
fread(void *restrict ptr, size_t size, size_t nobj, FILE *stream)
头文件: #include <stdio.h>
参数说明:第一个参数ptr是读出的数据存放的地址,也就是读出来的数据都存放在ptr指向的地址;第二个参数size为单个元素的大小,即由指针写入地址的数据大小,单位是字节;第三个参数nobj为元素个数,即要读取的数据大小为size的元素个素;第4个参数stream是fopen函数的返回值,也就是打开文件的的指针。
返回值:函数成功,返回读取的总数据元素个数,失败返回错误号。
这两个函数的一个应用实例是分别统计一篇英语文章中字母,空格和数字的多少。代码如下:
#include <stdio.h> #include <stdlib.h> #define MAX 1024 #include <string.h> int main(void) { FILE *fp; char buf[MAX]; int n; char *p; int buf1[MAX]; int letter, number, blank; fp = fopen("stu.txt", "rb"); if(fp == NULL) { perror("fail to open"); exit(1); } letter = 0; number = 0; blank = 0; while((n = fread(buf, sizeof(char), MAX-1, fp)) > 0) { buf[n] = '\0'; p = buf; while(*p != '\0') { if(('a' <= *p && *p <= 'z') || ('A' <= *p && *p <= 'Z')) letter++; if(*p == ' ') blank++; if('0' <= *p && *p <= '9') number++; p++; } } if(n == -1) { perror("fail to read"); exit(1); } printf("the letteris : %d \nthe number is %d\nthe blank is %d\n", letter, number, blank); fclose(fp); return 0; }
上边这一段代码实现的是统计一篇英语文章中字母,空格和数字的多少,敲完这段代码之后才明白txt文件中存储的都是字符串,即使是数字存储在txt文件中也是以字符串的形式来存储的,因此,如果想要把txt文件中的数组信息读出来就需要先读出字符串,然后将字符串转化成数字。到了这里思路才开始有一些明朗,接下来就是两件事,把数据按字符串的形式读出来,然后将字符串中的不同部分分别读出来,插入链表。
把文件中字符串读出来可以用fgets函数,一行一行的读出来,然后将每一行进行转换,按照上边统计字符个数那样,整数就是十位乘以10加上个位,但是后来发现我们存储的学生成绩是float形式的数据,不能用这种方法来读出来,当时也想过降低要求,把学生成绩都变成整型的就解决了这个问题,但是还是觉得既然题目这样要求肯定有解决的办法。现在一个主要的问题就是找到如何将字符串转化成float型。
最终找到了一些函数,atoi函数将字符串转换成整数、atof函数将字符串转换成浮点数。
atoi (const char * str) 函数
参数说明:参数str是要转换的字符串,也可以是字符数组。
返回值:成功返回转换的整型数字,失败返回0;
函数说明:atoi() 函数会扫描参数 str 字符串,跳过前面的空白字符(例如空格,tab缩进等),直到遇上数字或正负符号才开始做转换,而再遇到非数字或字符串结束时('\0')才结束转换,并将结果返回。
同理的atof函数可以将字符串转换成浮点数。这两个函数解决了如何将txt文件中读出来的字符串转化成浮点数的问题,解决了问题的一半,下一个目标就是如何将读出来的字符串分段,将学生ID的字符串、学生姓名的字符串和学生成绩的字符串分开,分别进行转换就可以了。
这个时候再用fgets函数不是很方便去解决这个问题了,这个时候又找到了一个函数fscanf函数。
fscanf函数可以从文本中读一个字符串到指定的数组中。从下面这段代码中可以更直观的了解fscanf这个函数。
#include <stdio.h> #include <string.h> struct node{ char a[20]; char b[20]; char c[20]; char d[20]; }; int main() { FILE *fp; struct node buf; fp=fopen("1.txt", "r"); fscanf(fp,"%[^ ] %[^ ] %[^ ] %s",buf.a,buf.b,buf.c,buf.d); printf("[%s][%s][%s][%s]\n",buf.a,buf.b,buf.c,buf.d); return 0 ; }
我们知道scanf的用法,非常严格能够读进去,能够读进去空格。fscanf是遇到空格的时候或者“,”的时候停止前边的放在一个字符串中。我们的文件中是用空格来进行间隔的,这样就好理解上边代码中的那一句话了 fscanf(fp,"%[^ ] %[^ ] %[^ ] %s",buf.a,buf.b,buf.c,buf.d); 这里%[ ],是扫描集的意思,%[^ ]其中^的意思就是当fscanf一个一个字符从文件读上来的时候如果遇到“ ”空格就会停下来,就会把前面读取的字符存到buf中,以此类推就可以分别得到4个字符串,这样就完成了将文件中不同的信息分别存放在不同的字符串的需求。
也就是用fopen打开文件,用fscanf函数分段读取出学生信息,用atof函数和atoi函数将字符串进行转换,这样就能得到符合插入链表要求的学生信息,然后再创建一个链表结点把这些信息写进去,加上循环就能够得到一个学生成绩的链表。
下面函数演示了如何从文件中将信息读出来,然后将信息写入结点,但是没有创建链表。下面的程序中要说明的一点是,在实际操作中发现用fscanf函数进行循环的时候,只能读出来第一行,换行之后读不出来了,在%[^ ]前加上换行符号“\n”之后不能读出来第一个字符串,直接从第二个字符串开始读,因此想到了在第一个字符串前加空格的办法,来使得程序换行后从第一个字符串开始读,文件中信息的存储格式如一开始文章的图片所示。
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <fcntl.h> #define MAX 10 typedef struct student * Student; static Student head; static int ID_count; struct student { Student next; int ID; char name[MAX]; float chgrade; float mathgrade; float avegrade; }; int main(void) { FILE *fp; int i; int j = 0; int n = 1 ; int ID; char name[MAX]; float chgrade; float mathgrade; float avegrade; char a[20]; char b[20]; char c[20]; char d[20]; char e[20]; char f[20]; fp = fopen("1.txt", "r"); if(fp == NULL) { printf("fail to fopen\n"); exit(1); } while(n != -1) { n = fscanf(fp, " %[^ ] %[^ ] %[^ ] %[^ ] %[^ ]", a, b, c, d, e); ID = atoi(a); for(i = 0; i < 10; i++) { name[i] = b[i]; } chgrade = atof(c); mathgrade = atof(d); avegrade = atof(e); printf("%d\n%s\n%.2f\n%.2f\n%.2f\n", ID, name, chgrade, mathgrade, avegrade); printf("n = %d\n", n); j++; } printf("%d\n", j); return 0; }