文件
13 文件
13.1 C 文件概述
文件是程序中一个重要的概念,所调文件一般是指存储在外部介质上数据的集合。一批数据是以文件的形式存放在外部介质(如磁盘)上的。操作系统是以文件为单位对数据进行管理的,也就是说,如果想找存在外部介质上的数据,必须先按文件名找到所指定的文件,然后再从该文件中读取数据。要向外部介质上存储数据也必须先建立一个文件(以文件名标识),才能向它输出数据。
C 语言把文件看作是一个字符(字节)的序列,即由一个一个字符(字节)的数据顺序组成。根据数据的组织形式,可以分为 ASCII 文件和二进制文件。ASCII 文件以称文本(text)文件,它的每一个字节放一个ASCII代码,代表一个字符。二进制文件是把内存中的数据按其在内存中的存储形式原样输出到磁盘上存放,如果有一个整数10000,在内存占2个字节,如果按ASCII码形式输出,则占5个字节,而按二进制形式输出,在磁盘上只占2个字节,用 ASCII 码形式输出与字符一一对应,一个字节代表一个字符,因而便于对字符进行逐个处理,也便于输出字符,但一般占存储空间较多,而且要花费转换时间(二进制形式与ASCII码间的转换)。用二进制形式输出值,可以节省外存空间和转换时间,但一个字节并不对应一个字符,不能直接输出字符形式,一般中间结果数据需要暂时保存在外存上以后又需要输入到内存的,常用二进制文件保存。
一个C 文件是一个字节流或二进制流。它把数据看作是一连串的字符(字节),而不考虑记录的界限。换句话说,C语言中文件并不是由记录组成。在C语言中对文件的存取是以字符(字节)为单位的。输入输出的数据流的开始和结束仅受程序控制而不受物理符号(如回车换行符)控制。也就是说,在输出时不会自动增加回车换行符作为记录结束的标志,输入时不以回车换行符作为记录的间隔(事实上C文件并不由记录构成)。我们把这种文件称为流式文件。C语言允许对文件取一个字符。这就是增加了处理的灵活性。
在过去使用的C版本(如UNIX系统下使用的C)有两种对文件处理的方法,缓冲文件系统,非缓冲文件系统,所谓缓冲文件系统是指系统自动地在内在区为第一个正在使用的文件名开辟一个缓冲区。从内在向磁盘输出数据必须先送到内存中的缓冲区,装满缓冲区后才一起送到磁盘去。如果从磁盘向内存读入数据,则一次从磁盘文件将一批数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(给程序变量)。
非缓冲文件系统,是指系统不自动开辟确定大小的缓冲区,而且程序为每个文件设定缓冲区。
在UNIX 系统下,用缓冲文件系统来处理文本文件,用非缓冲文件系统处理二进制文件。用缓冲文件系统进行的输入输出又称为高级磁盘输入输出,用非缓冲文件系统进行的输入输出又称为低级输入输出系统,ANSI C 标准决定不采用非缓冲文件系统,而只是采用缓冲文件系统。即既用缓冲文件系统处理文本文件,也用来处理二进制文件。也就是将缓冲文件系统扩充为可以处理二进制文件。
在 C 语言中,没有输入输出语句,对文件的读写都是用库函数来实现的,ANSI 规定了标准输入输出函数,用它们对文件进行读写。
13.2 文件类型指针
缓冲文件系统中,关键的概念是 文件指针。每个被使用的文件都在内存中开辟一个区,用来存放文件的有关信息(如文件名字,文件状态,文件当前位置)。这些信息是保存在一个结构体变量的。该结构体类型是由系统定义的,取名为 FILE 。Turbo c 在 stdio.h 文件中有以下的文件类型声明:
typedef sturct
{
short level;//缓冲区 满 或 空 的程度
unsigned flags;//文件状态标志
char fd;//文件描述符
unsigned char hold;//如无缓冲区不读取字符
short bsize;//缓冲区的大小
unsigned char *baffer;//数据缓冲区的位置
unsigned ar *curp;//指针,当前的指向
unsigned istemp;//临时文件指示器
short token;//用于有效性检查
}FILE;
有了结构体 FILE 类型后,可以用来定义若干个 FILE 类型的变量,以便存放若干个文件的信息。例如可以定义以下 FILE 类型的数组。
FILE f[5];
定义了一个结构体数据 f ,它有 5 个元素,可以用来存放 5 个文件信息。
可以定义文件型指针变量。如:
FILE *fp;
fp 是指一个指向 FILE 类型变量结构体的指针变量。可以使 fp 指向某一个文件的结构体变量,从而通过该结构体变量中的文件信息能够访问该文件。也就是说,通过文件指针变量能够找到与它相关的文件。如果有 n 个文件,一般应设 n 个指针变量(指向 FILE 类型结构体的指针变量),使它们分别指向 n 个文件,(指向存放该文件信息的结构体变量)以实现对文件的访问。
13.3 文件的打开与关闭
13.3.1 文件的打开 fopen 函数
ANSI C 规定了标准输入输出函数库,用 fopen() 函数来实现打开文件。fopen 函数的调用方式通常为
FILE *fp;
fp = fopen(文件名,使用文件方式);
例如:
fp = fopen("a1", "r");
它表示要打开名为 a1 的文件,使用文件方式为 读入(r 代表 read 即 读入),fopen 函数带回指向 a1 文件的指针并赋给 fp,这样 fp 就和文件 a1 相联系了,或者说,fp 指向 a1 文件。可以看出,在打开一个文件时,通知给编译系统以下 3 个信息:1 需要打开的文件名,也就是准备访问的文件的名字。2 使用文件的方式(读还是写)3 让哪一个指向变量指向被打开的文件。
使用文件方式表
文件使用方式 含义
r 只读 为输入打开一个文本文件
w 只写 为输出打开一个文本文件
a 追加 向文本文件尾增加数据
rb 只读 为读取打开一个二进制文件
wb 只写 为写入打开一个二进制文件
ab 追加 向二进制文件尾增加数据
r+ 读写 为读写打开一个文本文件
w+ 读写 为读写建立一个新的文本文件
a+ 读写 为读写打开一个文本文件
rb+ 读写 为读写打开一个新的二进制文件
wb+ 读写 为读写建立一个新的二进制文件
ab+ 读写 为读写打开一个二进制文件
说明:
(1)用 r 方式打开的文件只能用于向计算机输入而不能用作向该文件输出数据(写入数据),而且该文件应该已存在,不能用 r 方式打开一个并不存在的文件,否则出错。
(2)用 w 方式打开的文件只能用于向该文件写数据(即输出到文件),而不能用来向计算机输入。如果原来不存在该文件,则在打开时新建立一个以指定的名字命名的文件。如果原来已存在一个以该文件名命名的文件,则在打开时将该文件删去,然后再新建立一个新的文件。
(3)如果希望向文件末尾添加新的数据(不希望删除原有数据)则应该用 a 方式打开。因此该文件必须已存在,否则将得到出错信息。打开时,位置指针移到文件末尾。
(4)用 r+, w+, a+ 方式打开的文件既可以用来输入数据,也可以用来输出数据,用 r+ 方式时,该文件应该已存在,以便能向计算机输入数据。用 w+ 方式则新建立一个文件,先向此文件写数据,然后可以读此文件中的数据。用 a+ 方式打开文件,原来的文件不被删除,位置指针移动到文件末尾,可以添加,也可以读。
(5)如果不能实现 打开 任务, fopen 函数将会带回一个出错信息。出错的原因可能是用 r 方式打开一个并不存在的文件,磁盘出故障,磁盘已满无法建立新文件等。此时 fopen 函数将带回一个空指针值 NULL(NULL 在 stdio.h 文件中定义为0)。
常用下面的方式打开一个文件:
if((fp=fopen("flie1","r")) == NULL)
{
printf("can not open this file.\n");
exit(0);
}
即先检查打开的操作有否出错,如果出错就在终端输出出错信息,exit 函数的作用是关闭所有文件,终止正在调用的过程,待用户检查出错误,修改后再运行。
(6)用以上方式可以打开文本文件或是二进制文件,这是 ANSI C 的规定,用同一种缓冲文件系统来处理文件和二进制文件,但目前使用的有些C 编译系统可能不匠心有这此功能。(例如有的只能用 r, w, a 方式),有的C版本不用 r+, w+, a+ 而用 rw, wr, ar等。
(7)在向计算机输入文件文件时,将回车换行符转换为一个换行符,在输出时把换行符转换成为回车的换行两个字符,在用二进制文件时,不进行这种转换,在内存中的数据形式与输出到外部文件中的数据形式完全一致,一一对应。
(8)在程序开始运行时,系统自动打开3 个标准文件,标准输入,标准输出,标准出错,通常这3个文件都与终端相联系,因此以前我们所用到的从终端输入或输出都不需要打开终端文件。系统自动定义了 3 个文件指针 stdin, stdout, stderr 分别指向终端输入,终端输出,标准出错输出(也从终端输出)。如果程序中指定要从 stdin 所指的文件输入数据,就是指从终端键盘输入数据。
13.3.2 文件的关闭 fclose 函数
在使用完一个文件后应该关闭它,以防止它再误用,关闭就是指文件指针变量不指向该文件,也就是文件指针变量与文件脱钩,此后不能再通过该指针对原来与其相联系的文件进行读写操作。除非再次打开,使该指针变量重新指向该文件。
用 fclose 函数关闭文件,fclose 函数调用的一般形式为
fclose(文件指针);
例如:
fclose(fp);
应该养成在程序终止之前关闭所有文件的习惯,如果不关闭文件将丢失数据。因为如前所述,在向文件写数据时,是先将数据输到缓冲区,待缓冲区充满后才正式输出给文件,如果当数据未充满缓冲区,而程序结束运行,就会将缓冲区中的数据丢失,用 fclose 函数关闭文件,可以避免这个问题,它先把缓冲区中的数据输出到磁盘文件,然后才释放文件指针变量。
fclose 函数也带回一个值,当顺利地执行了关闭操作,则返回值为 0 ,否则返回 EOF (-1) 。可以用 ferror 函数来测试。
13.4 文件的读写
13.4.1 fputc 函数和 fgetc 函数(putc 函数和 getc 函数)
1. fputc 函数
把一个字符写到磁盘文件上去。其一般调用形式为
fputc(ch, fp);
其中 ch 是要输出的字符,它可以是一个字符常量,也可以是一个字符变量,fp 是文件指针变量,fputc(ch, fp); 函数的作用是将字符(ch 的值)输出到 fp 所指向的文件中去。fputc 函数也带回一个值,如果输出成功则返回值就是输出的字符。如果输出失败,则返回一个 EOF(-1),EOF 是在 stdio.h 文件中定义的符号常量值为 -1。
2. fgetc 函数
从指定的文件读入一个字符,该文件必须是以读或写方式打开的,fgetc 函数的调用形式为
ch = fgetc(fp);
fp 为文件型指针变量,ch 为字符变量。fgetc 函数带回一个字符,赋给 ch。如果在执行 fgetc 函数读字符时遇到文件结束符,函数返回一个文件结束标志 EOF(-1)。如果想从一个磁盘文件顺序读入字符并在屏幕上显示出来 ,可以:
ch = fgetc(fp);
while(ch != EOF)
{
putchar(ch);
ch = fgetc(fp);
}
注意:EOF 不是可输出字符,因此不能在屏幕上显示。由于字符的 ASCII 可能出现,-1 ,因此 EOF 定义为 -1 是合适的。当读入的字符值等于 -1 (即 EOF )时,表示读入的已不是正常字符而是文件结束符。但以上只适用于读文本文件的情况。现在ANSI C 已允许用缓冲文件系统处理二进制文件,而读入一个字节中的二进制数据的值可是 -1,而这又恰好是 EOF 的值。这就是出现了需要读入朋用数据而却被处理为 文件结束的情况,为了解决这个问题,ANSI C 提供一个 feof 函数来判断文件是否真的结束。feof(fp) 用来测试 fp 所指向的文件当前状态是否 文件结束。如果是文件结束,函数 feof(fp) 的值为1,否则为0。
如果想顺序读入一个二进制文件中的数据可以用
while( ! feof(fp))
{
c = fgetc(fp);
...
}
当未遇文件结束,fefo(fp) 的值为 0, !feof(fp) 为 1,读入一个字节的数据赋给整形变量c ,并接着对其进行所需的处理,直到遇文件结束,fefo(fp) 值为 1,!feof(fp) 值为0,不再执行 while 循环。
这种方法也适用于文本文件。
3 fputc 和 fgetc 函数使用举例
#include <stdio.h>
void main()
{
FILE *fp;
char ch, filename[10];
scnaf("%s", filename);
if((fp = fopen(filename, "w")) == NULL)
{
printf("can not open file. \n");
exit(0);
}
ch = getchar();
while( ch != '#')
{
fputc(ch, fp);
putchar(ch);
ch = getchar();
}
fclose(fp);
}
将一个磁盘文件中的信息复制到另一个磁盘文件中
#include <stdio.h>
void main()
{
FILE *in, *out;
char ch, infile[10], outfile[10];
printf("Enter the infile name: \n");
scanf("%s", infile);
printf("Enter the outfile name:\n");
scnaf("%s", outfile);
if((in = fopen(infile, "r")) == NULL)
{
printf("can not open infile.\n");
exit(0):
}
if((out= fopen(outfile, "w")) == NULL){
printf("can not open outfile.\n");
exit(0):
}
while(!feof(in))
fputc(fgetc(in), out);
fclose(in);
fclose(out);
}
13.4.2 fread 函数和 fwrite 函数
用 fgetc 和 fputc 函数可以用来读写文件中的一字符。但是常常要求一次读入一给数据(例如,一个实数或一个结构体变量的值),ANSI C 标准提出设置两个函数 fread 和 fwrite,用来读写一个数据块,它们的一般形式为
fread(buffer, size, count, fp);
fwrite(buffer, size, count, fp);
期中:
buffer,是一个指针,对 fread 来说,它是读入数据的存放的地址。对 fwrite 来说,是要输出数据的地址(以上指的是起始地址)。
size ,是读写的字节数。
count,要进行读写多少个 size 个字节数据项。
fp,文件型指针变量,
如果文件以二进制形式打开,用 fread 和 fwrite 函数可以读写任何类型的信息,如:
fread(f, 4, 2, fp);
其中 f 是一个实型数组名,一个实型变量占 4 个字节,这个函数从 fp 所指向的文件读入 2 次(每次 4 个字节)数据,存储到数组 f 中。
如果有一下如下的结构体类型:
struct student_type
{
char name[10];
int num;
int age;
char addr[30];
}stud[40];
结构体数组 stud 有 40 个元素,每一个元素用来存放一个学生的数据,假设学生的数据已存放在磁盘文件中,可以用下面的 for 语句和 fread 函数读入 40 个学生数据:
for(i-0;i<40;i++)
fread(&stud[i], sizeof(struct student_type), 1, fp);
同样,以下 for 语句和 fwrite 函数可以将内存中的学生数据输出到磁盘文件中去:
for(i=0;i<40;i++)
fwrite(&stud[i], sizeof(struct student_type), 1, fp);
如果 fread 和 fwrite 调用成功,则函数返回值为 count 的值,即输入或输出数据项的完整个数。
程序例子:
从键盘输入4个学生的有关数据,然后把它们转存到磁盘文件上去。
#include <stdio.h>
#include <stdlib.h>
#define SIZE 4
struct student_type
{
char name[10];
int num;
int age;
char addr[30];
}stud[SIZE];
void save()
{
FILE *fp;
int i;
if((fp = fopen("stu_list", "wb")) == NULL)
{
printf("can not open file\n");
return;
}
for(i=0;i<SIZE;i++)
{
if((fwrite(&stud[i], sizeof(struct student_type), 1, fp)) != 1)
printf("file write error\n");
}
}
void main()
{
int i;
for(i=0;i<SIZE;i++)
scanf("%s%d%d%s", stud[i].name, &stud[i].num, &stud[i].age, stud[i].addr);
save();
system("pause");
}
以下程序读取出显示出来
void read()
{
FILE *fp;
int i;
if((fp = fopen("stu_list", "rb")) == NULL)
{
printf("can not open file\n");
return;
}
for(i=0;i<SIZE;i++)
{
if((fread(&stud[i], sizeof(struct student_type), 1, fp)) != 1)
printf("file write error\n");
else
printf("%s %d %d %s\n",stud[i].name, stud[i].num, stud[i].age, stud[i].addr);
}
}
注意输入输出数据的状况,从键盘输入4个学生的数据是 ASCII 码,也就是文本文件,在送到计算机内存时,回车和换行符转换成一个换行符。再从内存以 wb 方式(二进制写)输出到 stu_list 文件,此时不发生字符转换,按内存中存储形式原样输出到磁盘文件上。在上面的程序中,又用到 fread 函数从 stu_list 文件向内存读入数据,注意此时用的是 rb 方式即二进制方式,数据按原样输入,也不发生字符转换,也就是这时候内存的数据恢复到第一个程序向 stu_list 输出前的情况。最后在验证程序中,用 printf 函数输出到屏幕,printf 是格式输出函数,输出 ASCII 码,在屏幕上显示字符,换行符又转换为回车加换行符。
如果企图从 stu_list 文件中以 r 方式读入数据就会出错。
fread 和 fwrite 函数一般用于二进制文件的输入输出。因为它们是按数据块的长度来处理输入输出的,在字符发生转换的情况下很可能出现与原设想的情况不同。
例如写成如下
fread(&stud[i[, sizeof(struct student_type)), 1, stdin);
企图从终端键盘输入数据,这在语法上并不存在错误,编译能通过。如果用以下形式输入数据:
Zhang 1001 10 room_101
由于 fread 函数要求一次输入 29 个字节(而不问这些字节的内容),因此输入数据中的空格也作为输入数据而不作为数据间的分隔符了。连空格也存储到 stud[i]中了,显然是不对的。
从磁盘文件中读二进制数据
void load()
{
FILE *fp;
int i;
if((fp = fopen("stu_list", "rb")) == NULL)
{
printf("can not open infile\n");
return;
}
for(i=0;i<SIZE; i++)
{
if((fread(&stud[i], sizeof(struct student_type), 1, fp)) != 1)
{
if(feof(fp)) return;
printf("file read error \n");
}
}
}
13.4.3 fprintf 函数和 fscanf 函数
fprintf 函数 fscanf 函数与 printf 函数 scanf 函数作用相仿,都是格式化读写函数。只有一点不同,fprintf 和 fscanf 函数的读写对象不是终端而是磁盘文件,它们的一般调用方式为
fprintf(文件指针,格式字符串,输出表列);
fscanf(文件指针,格式字符串,输入表列);
例如:
fprintf(fp, "%d, %d6.2f", i, t);
它的作用是将整型变量i 和实型变量 t 的值按 %d 和 %6.2f 的格式输出到 fp 指向的文件里,如果 i=3, t=4.5 则输出到磁盘文件上的是以下的字符串:
3, 4.5
同样,用以下 fscanf 函数可以从磁盘文件上读入 ASCII 字符:
fscanf(fp, "%d, %f", &i, &t);
磁盘文件如果有以下字符:
3, 4.5
则将磁盘文件中的数据 3 送给变量 i,4.5 送给变量 t。
同 fprintf 和 fscanf 函数对磁盘文件读写,使用方便,容易理解,但由于在输入时要将ASCII 转换为二进制形式,在输出时又要将二进制形式转换成字符,花费时间比较多。因此,在内存与磁盘频繁交换数据的情况下,最好不用 fprintf fscanf 函数,而用 fread 和 fwrite 函数。
13.4.4 其它读写函数
1. putw 和 getw 函数
大多数 C 编译系统都提供另外两个函数,putw 和 getw ,用来对磁盘文件读写一个字(整数)。例如:
putw(10, fp);
它的作用是将整数 10 输出到 fp 指向的文件。而
i = getw(fp);
的作用是从磁盘文件读一个整数到内存,赋给整型变量 i。
如果所用的 C编译的库函数不包括 putw 和 getw 函数,可以自己定义该函数。putw 函数如下。
int putw(int i, FILE *fp)
{
char *s;
s = &i;
putc(s[0], fp);
putc(s[1], fp);
return i;
}
当调用 putw 函数时,如果用 putw(10, fp) 语句,形参 i 得到实参传来的值 10,在 putw 函数中将 i 的地址赋予指针变量 s, 而 s 是指向字符变量的指针变量,因此 s 指向 i 的第1 个字节,s + 1 指向 i 的第2个字节。由于 *(s + 0) 就是 s[0],*(s + 1) 就是 s[1],因此 s[0], s[1] 分别对应 i 的第 1 字节和第 2 字节。顺序输出 s[0], s[1] 就相当于输出了 i 的两个字节中的内容。
getw 函数如下:
int getw(FILE *fp)
{
char *s;
int i;
s = &i;
s = &i;
s[0] = getc(fp);
s[1] = getc(fp);
return i;
}
2. 读写其他类型数据
如果 ANSI C 提供的 fread 和 fwrite 函数,读写任何类型数据都是十分方便的。如果所用的系统不提供这两个函数,用户只好自己定义所需函数,例如可以定义一个向磁盘文件写一个实数(用二进制方式)的函数 putfloat
void putfolat(float num, FILE *fp)
{
char *s;
int count;
s = #
for(count = 0; count <4; count ++)
putc(s[count], fp);
}
3. fgets 函数与 fputs 函数
fgets 的作用是从指定文件读入一个字符串。如:
fgets(str, n, fp);
n 为要求得到的字符,但只从 fp 指向的文件输入 n - 1 个字符,然后在最后加一个 \0 字符,因此得到字符串共有 n 个字符,把它们放到字符数组 str 中。如果在读完 n - 1 个字符之前遇到换行符或是 EOF ,读入即结束。fgets 函数返回值为 str 的首地址。
fputs 函数作用是向指定的文件输出一个字符串。如:
fputs("china", fp);
把字符串 china 输出到 fp 指向的文件,fputs 函数中第一个参数可以是字符串变量,字符数组名或字符型指针。字符串末尾的 \0 不输出。若输出成功,函数值为 0,失败时为 EOF。
13.5 文件的定位
文件中有一个位置指针,指向当前读写的位置,如果顺序读写一个文件,每次读写一个字符,则读写完一个字符后,该位置指针自动移动指向下一个字符位置。如果想改变这样的规律,强制使位置指针指向其它指定的位置,可以用有关函数。
13.5.1 rewind 函数
rewind 函数的作用是使位置指针重新返回文件的开关。此函数没有返回值。
#include <stdio.h>
void main()
{
FILE *fp1, *fp2;
fp1 = fopen("file1.c", "r");
fp2 = fopen("file2.c", "w");
while(!feof(fp1)) putchar(fgetc(fp1));
rewind(fp1);
while(!feof(fp1)) fputc(fgetc(fp1), fp2);
fclose(fp1);
fclose(fp2);
}
13.5.2 fseek 函数和随机读写
对流式文件可以进行顺序读写,也可以进行随机读写。关键在于控制文件的位置指针,如果位置指针是按字节位置顺序移动的,就是顺序读写。如果能将位置指针按需要移动到任意位置,就可以实现随机读写,所谓随机读写,是指读写完上一个字符后,并不一定要读写其后续的字符,而可以读写文件中任意所需的字符(字节)。
用 fseek 函数可以实现改变文件的位置指针。
fseek 函数的调用形式为
fseek(文件类型指针,位移置,起始点);
起始点,用 0, 1 或 2 代替,0 代表 文件开始 1 为当前位置 2 为文件末尾。ANSI C 标准指定的名字
起始点 名字 用数据代表
文件开始 SEEK_SET 0
文件当前位置 SEEK_CUR 1
文件末尾 SEEK_END 2
位移量 指以 起始点 为基点,向前移动的字节数, ANSI C 和大多数 C 版本要求位移量是 long 型数据。这样当文件的长度大于 64K 时不致出问题,ANSI C 标准规定在数字末尾加一个字母 L 就表示是 long 型。
fseek 函数一般用于二进制文件,因为文本文件要发生字符转换,计算位置时往往会发生混乱。
下面是 fseek 函数调用的例子:
fseek(fp, 100L, 0);// 将位置指针移到离文件头100个字节处
fseek(fp, 50l, 1);// 将位置指针移到离当然位置 50个字节处
fseek(fp, -10L, 2);// 将位置指针从文件末尾处向后退10 个字节
#include <stdio.h>
struct studnt_type
{
char name[10];
int num;
int age;
char sex;
}stud[10];
void main()
{
int i;
FILE *fp;
if((fp = fopen("stud_dat", "rb")) == NULL)
{
printf("can not open file \n");
exit(0);
}
for(i=0i<10;i++)
{
fseek(fp, i * sizeof(struct student_type), 0);
fread(&stud[1], sizeof(struct student_type), 1, fp);
printf("%s, %d, %d, %c\n", sutd[1].name, stud[i].num, sutd[i].age, stud[i].sex);
}
fclose(fp);
}
13.5.3 ftell 函数
ftell 函数作用是得到流式文件中的当前位置,用相对于文件开关的位移量来表示,由于文件中的位置指针经常移动,人们往往不容易知道其当前位置。用 ftell 函数可以得到当前位置。如果 ftell 函数返回值为 -1L,表示出错。例如:
i = ftell(fp);
if(i == -1L) printf("error\n");
变量 i 存放当前位置,如调用函数出错(如不存在些文件)则输出 error。
13.6 出错的检测
13.6.1 ferror 函数
在调用各种输入输出函数(如 fputc, fgetc, fread, fwrite 等)时,如果出现错误,除了函数返回值所反映外,还可以用 freeor 函数来检查。它的一般调用形式为
ferror(fp);
如果 ferror 返回值为 0 (假),表示末出错,如果返回一个非零值,表示出错。应该注意,对同一个文件每一次调用输入输出函数,均产生一个新的 ferror 函数值,因此,应当在调用一个输入输出函数后立即检查 ferror 函数的值,否则信息会丢失。
在执行 fopen 函数时,ferror 函数的初始值自动置为 0。
13.6.2 clearerr 函数
它的作用是使文件错误标志和文件结束标志置为 0,假设在调用一个输入输出函数时出现错误,ferror 函数值为一个非零值。在调用 clearerr(fp) 后,ferror(fp) 的值变成 0。只要出现错误标志,就一直保留,直到对同一文件调用 clearerr 函数或 rewind 函数。或任何其它一个输入输出函数。
13.7 文件输入输出小结
分类 函数名 功能
打开文件 fopen() 打开文件
关闭文件 fclose() 关闭文件
文件定位 fseek() 改变文件位置的指针位置
rewind() 使文件位置指针重新置于文件开头
ftell() 返回文件位置指针的当前值
文件读写 fgetc(), getc() 从指定文件取得一个字符
fputc(), putc() 把字符输出到指定文件
fgets() 从指定文件读取字符串
fputs() 把字符串输出到指定文件
getw() 从指定文件读取一个字
putw() 把一个字输出到指定文件
fread() 从指定文件中读取数据项
fwrite() 把数据项写到指定文件
fscanf() 从指定文件按格式输入数据
fprintf() 按指定格式将数据写到指定文件中
文件状态 feof() 若到文件末尾函数返回 真 非零
ferror() 若对文件操作出错,函数返回 真 非零
clearerr() 使ferror 和 feof 函数值置零