对文件的读和写是最常用的文件操作。在C语言中提供了多种文件读写的函数:
- 字符读写函数 :fgetc 和 fputc
- 字符串读写函数:fgets 和 fputs
- 数据块读写函数:freed 和 fwrite
- 格式化读写函数:fscanf 和 fprinf
使用以上函数都要求包含头文件 stdio.h。
下面分别予以介绍,字符读写函数是以字符(字节)为单位的读写函数。 每次可从文件读出或向文件写入一个字符。
一. 读字符函数 fgetc
fgetc 函数的功能是从指定的文件中读一个字符,函数调用的形式为:
字符变量=fgetc(文件指针);
例如:
ch=fgetc(fp);
其意义是从打开的文件 fp 中读取一个字符并送入 ch 中。 对于 fgetc 函数的使用有以下几点说明:
1) 在 fgetc 函数调用中,读取的文件必须是以读或读写方式打开的。
2) 读取字符的结果也可以不向字符变量赋值, 例如:
fgetc(fp);
但是读出的字符不能保存。
3) 在文件内部有一个位置指针。用来指向文件的当前读写字节。在文件打开时,该指针总是指向文件的第一个字节。使用 fgetc 函数后,该位置指针将向后移动一个字节。 因此可连续多次使用 fgetc 函数,读取多个字符。应注意文件指针和文件内部的位置指针不是一回事。文件指针是指向整个文件的,须在程序中定义说明,只要不重新赋值,文件指针的值是不变的。文件内部的位置指针用以指示文件内部的当前读写位置,每读写一次,该指针均向后移动,它不需在程序中定义说明,而是由系统自动设置的。
fgetc函数的应用举例。我在我计算机的C盘/0Clanguage/File 目录下创建一个文件,goodguy.txt
内容为:
I am a good guy and you are a good guy too.
We are both good guys.
例1:利用fgetc读取goodguys.txt 文件
#include<stdio.h> main() { FILE *fp; char ch; if((fp=fopen("c:\\0Clanguage\\files\\goodguys.txt","rt"))==NULL) { printf("\nCannot open file strike any key exit!"); getch(); exit(1); } ch=fgetc(fp); while(ch!=EOF) { putchar(ch); ch=fgetc(fp); } fclose(fp); }
本例程序的功能是从文件中逐个读取字符,在屏幕上显示。程序定义了文件指针 fp,以
读文本文件方式打开文件“c:\\0Clanguage\\files\\goodguys.txt”,并使 fp 指向该文件。
如打开文件出错,给出提示并退出程序。程序第 12 行先读出一个字符,然后进入循环,只要读出的字符不是文件结束标志(每个文件末有一结束标志 EOF)就把该字符显示在屏幕上,再读入下一字符。每读一次,文件内部的位置指针向后移动一个字符,文件结束时,该指针指向 EOF。执行本程序将显示整个文件。
二. 写字符函数 fputc
fputc 函数的功能是把一个字符写入指定的文件中,函数调用的形式为:
fputc(字符量,文件指针);
其中,待写入的字符量可以是字符常量或变量,例如:
fputc('a',fp);
其意义是把字符 a 写入 fp 所指向的文件中。
对于 fputc 函数的使用也要说明几点:
1) 被写入的文件可以用写、读写、追加方式打开,用写或读写方式打开一个已存在的文件时将清除原有的文件内容,写入字符从文件首开始。如需保留原有文件内容,希望写入的字符以文件末开始存放,必须以追加方式打开文件。被写入的文件若不存在,则创建该文件。
2) 每写入一个字符,文件内部位置指针向后移动一个字节。
3) fputc 函数有一个返回值,如写入成功则返回写入的字符,否则返回一个 EOF。可用此来判断写入是否成功。
例2.从键盘输入一行字符,写入一个文件,再把该文件内容读出显示在屏幕上
#include<stdio.h> main() { FILE *fp; char ch; if((fp=fopen("c:\\0Clanguage\\files\\goodguys.txt","at+"))==NULL) { printf("\nCannot open file strike any key exit!"); getch(); exit(1); } printf("input a string:\n"); ch=getchar(); while (ch!='\n') { fputc(ch,fp); ch=getchar(); } rewind(fp); ch=fgetc(fp); while(ch!=EOF) { putchar(ch); ch=fgetc(fp); } printf("\n"); fclose(fp); }
程序中第 6 行以读写文本文件方式打开文件 string。程序第 13 行从键盘读入一个字符后进入循环,当读入字符不为回车符时,则把该字符写入文件之中,然后继续从键盘读入下一字符。
每输入一个字符,文件内部位置指针向后移动一个字节。写入完毕,该指针已指向文件末。
如要把文件从头读出,须把指针移向文件头,程序第 19 行 rewind 函数用于把 fp所指文件的内部位置指针移到文件头。 第 20 至 25 行用于读出文件中的一行内容。
例3.向文件输出字符并打印出来
#include <stdio.h> int main() { int i; FILE * fptr; char ch; char fn[50]; char str[] = "riscV Rocks\n"; fptr = fopen("fputc_test.txt", "w+"); // "w" defines "writing mode" for (i = 0; str[i] != '\n'; i++) { /* write to file using fputc() function */ fputc(str[i], fptr); } rewind(fptr); ch=fgetc(fptr); while(ch!=EOF) { putchar(ch); ch=fgetc(fptr); } printf("\n"); fclose(fptr); return 0; }
上面的程序将一个字符写入fputc_test.txt文件,直到到达下一行符号“ \ n”,这表明该句子已成功写入。 该过程是获取数组的每个字符并将其写入文件。
以上示例中,利用fopen函数产生文件fputc_test.txt,并进入写入模式。然后利用fputc写入内容。利用fclose关闭文件指针。
三. 读字符串函数 fgets
函数的功能是从指定的文件中读一个字符串到字符数组中,函数调用的形式为:
fgets(字符数组名,n,文件指针);
其中的 n 是一个正整数。表示从文件中读出的字符串不超过 n-1 个字符。在读入的最后一个字符后加上串结束标志’\0’。
例如:
f
gets(str,n,fp);
的意义是从 fp 所指的文件中读出 n-1 个字符送入字符数组 str 中。
例4 从 string 文件中读入一个含 10 个字符的字符串。
#include<stdio.h> main() { FILE *fp; char str[11]; if((fp=fopen("c:\\0Clanguage\\files\\goodguys.txt","rt"))==NULL) { printf("\nCannot open file strike any key exit!"); getch(); exit(1); } fgets(str,11,fp); printf("\n%s\n",str); fclose(fp); }
本例定义了一个字符数组 str 共 11 个字节,在以读文本文件方式打开文件 string 后,从中读出 10 个字符送入 str 数组,在数组最后一个单元内将加上’\0’,然后在屏幕上显示输出 str 数组。
输出的十个字符正是例 goodguy.txt 文本文件的前十个字符。
对 fgets 函数有两点说明:
- 在读出 n-1 个字符之前,如遇到了换行符或 EOF,则读出结束;
- fgets 函数也有返回值,其返回值是字符数组的首地址。
四.写字符串函数 fputs
fputs 函数的功能是向指定的文件写入一个字符串,其调用形式为:
fputs(字符串,文件指针);
其中字符串可以是字符串常量,也可以是字符数组名,或指针变量,例如:
fputs(“abcd“,fp);
其意义是把字符串“abcd”写入 fp 所指的文件之中。
例5. goodguy.txt追加一个字符串
#include<stdio.h> main() { FILE *fp; char ch,st[20]; if((fp=fopen("c:\\0Clanguage\\files\\goodguys.txt","at+"))==NULL) { printf("\nCannot open file strike any key exit!"); getch(); exit(1); } printf("input a string:\n"); scanf("%s",st); fputs(st,fp); rewind(fp); ch=fgetc(fp); while(ch!=EOF) { putchar(ch); ch=fgetc(fp); } printf("\n"); fclose(fp); }
五. 数据块读写函数 fread 和 fwtrite
C语言还提供了用于整块数据的读写函数。可用来读写一组数据,如一个数组元素,一个结构变量的值等。
读数据块函数调用的一般形式为:
fread(buffer,size,count,fp);
写数据块函数调用的一般形式为:
fwrite(buffer,size,count,fp);
其中:
buffer 是一个指针,在 fread 函数中,它表示存放输入数据的首地址。在 fwrite 函数中,它表示存放输出数据的首地址。
- size 表示数据块的字节数;
- count 表示要读写的数据块块数;
- fp 表示文件指针。例如:
fread(fa,4,5,fp);
其意义是从 fp 所指的文件中,每次读 4 个字节(一个实数)送入实数组 fa 中,连续读 5 次,即读 5 个实数到 fa 中。
例6.数据块读取文件内容的联系
#include<stdio.h> struct stu { char name[10]; int num; int age; char addr[15]; }boya[2],boyb[2],*pp,*qq; main() { FILE *fp; char ch; int i; pp=boya; qq=boyb; if((fp=fopen("stu_list.txt","wb+"))==NULL) { printf("Cannot open file strike any key exit!"); getch(); exit(1); } printf("\ninput data\n"); for(i=0;i<2;i++,pp++) scanf("%s%d%d%s",pp->name,&pp->num,&pp->age,pp->addr); pp=boya; fwrite(pp,sizeof(struct stu),2,fp); rewind(fp); fread(qq,sizeof(struct stu),2,fp); printf("\n\nname\tnumber age addr\n"); for(i=0;i<2;i++,qq++) printf("%s\t%5d%7d %s\n",qq->name,qq->num,qq->age,qq->addr); fclose(fp); }