在C语言中用一个指针变量指向一个文件, 这个指针称为文件指针。通过文件指针就可对它所指的文件进行各种操作。 定义说明文件指针的一般形式为: FILE* 指针变量标识符; 其中FILE应为大写,它实际上是由系统定义的一个结构(如下图, 该结构中含有文件名、文件状态和文件当前位置等信息)。
#ifndef _FILE_DEFINED
struct _iobuf {
char *_ptr;
int _cnt;
char *_base;
int _flag;
int _file;
int _charbuf;
int _bufsiz;
char *_tmpfname;
};
typedef struct _iobuf FILE;
#define _FILE_DEFINED
#endif /* _FILE_DEFINED */
在编写源程序时不必关心FILE结构的细节。例如:FILE *fp; 表示fp是指向FILE结构的指针变量,通过fp 即可找存放某个文件信息的结构变量,然后按结构变量提供的信息找到该文件, 实施对文件的操作。习惯上也笼统地把fp称为指向一个文件的指针。在进行读写文件操作之前要先打开,使用完毕要关闭。 所谓打开文件,实际上是建立文件的各种有关信息, 并使文件指针指向该文件,以便进行其它操作。关闭文件则断开指针与文件之间的联系,也就禁止再对该文件进行操作。
在C语言中,文件操作都是由库函数来完成的。 在本章内将介绍主要的文件操作函数如下:
打开关闭文件函数:FILE *fopen(char *filename, char *mode) | int fclose(FILE *fp)
字符 读写函数:int fgetc(FILE *fp) | int fputc(int ch, file *fp)
字符串读写函数:char *fgets(char *str, int num, FILE *fp) | int fputs(char *str, FILE *fp)
数据块读写函数:int fread(void *buf, int size, int count, FILE *fp) | int fwrite(void *buf, int size, int count, FILE *fp)格式化读写函数:int fscanf(FILE *fp, char *format,…) | int fprintf(FILE *fp, char *format,…)随机读写函数:int fseek(FILE *fp, long offset, int origin) | void rewind(FILE *fp)
文件检测函数:int feof(FILE *fp) | int ferror(FILE *fp) | void clearerr(FILE *fp)
一、打开文件操作:FILE *fopen(char *filename, char *mode)
filename: 采用绝对或相对路径的目标文件名
mode: 文件的类型和操作要求
返回值: 目标文件指针或空指针值NULL(打开异常时)
文件类型 :t (text): 文本文件(可省略不写); b (banary): 二进制文件
从文件编码的方式来看,文件可分为ASCII码文件和二进制码文件两种。
ASCII文件也称为文本文件,这种文件在磁盘中存放时每个字符对应一个字节,用于存放对应的ASCII码。ASCII码文件可在屏幕上按字符显示。二进制文件是按二进制的编码方式来存放文件的。
二进制文件虽然也可在屏幕上显示,但其内容无法读懂。
C系统在处理这些文件时,并不区分类型,都看成是字符流,按字节进行处理。输入输出字符流的开始和结束只由程序控制而不受物理符号(如回车符)的控制。 因此也把这种文件称作“流式文件”。把一个文本文件读入内存时,要将ASCII码转换成二进制码, 而把文件以文本方式写入磁盘时,也要把二进制码转换成ASCII码,因此文本文件的读写要花费较多的转换时间。对二进制文件的读写不存在这种转换。
文件操作类型:r (read): 读【目标文件必须存在,否则报错】
w (write): 写【目标不存在时自动创建】a (append): 追加【目标文件必须存在,否则报错】+ : 读和写
操作类型组合方式
操 作 说 明
"rt"
只读打开一个文本文件,只允许读数据
"wt"
只写打开或建立一个文本文件,只允许写数据
"at"
追加打开一个文本文件,并在文件末尾写数据
"rb"
只读打开一个二进制文件,只允许读数据
"wb"
只写打开或建立一个二进制文件,只允许写数据
"ab"
追加打开一个二进制文件,并在文件末尾写数据
"rt+"
读写打开一个文本文件,允许读和写
"wt+"
读写打开或建立一个文本文件,允许读写
"at+"
读写打开一个文本文件,允许读,或在文件末追加数据
"rb+"
读写打开一个二进制文件,允许读和写
"wb+"
读写打开或建立一个二进制文件,允许读和写
"ab+"
读写打开一个二进制文件,允许读,或在文件末追加数据
实例如下:
FILE *fp;
fp=("file a","r"); //打开当前目录下文件file a, 只允许进行“读”操作,并使fp指向该文件。
FILE *fphzk
fphzk=("c://hzk16',"rb"); //打开C驱动器磁盘的根目录下的文件hzk16, 这是一个二进制文件,只允许按二进制方式进行读操作
二、关闭文件操作:int fclose(FILE *fp)
fp: 待关闭文件的文件指针返回值: 0(正常关闭),非0(关闭异常)
三、 读字符函数: int fgetc(FILE *fp)
fp: 待读文件的文件指针返回值: 读出字符的ASCII码或EOF(文件结束时)
fgetc函数的功能是从指定的文件中读一个字符。在文件内部有一个位置指针。用来指向文件的当前读写字节。在文件打开时,该指针总是指向文件的第一个字节。使用fgetc 函数后, 该位置指针将向后移动一个字节。文件结束时,该指针指向EOF, 因此可连续多次使用fgetc函数,读取多个字符直至遇到EOF为止。 应注意文件指针和文件内部的位置指针不是一回事。文件指针是指向整个文件的,须在程序中定义说明,只要不重新赋值,文件指针的值是不变的。文件内部的位置指针用以指示文件内部的当前读写位置,每读写一次,该指针均向后移动,它不需在程序中定义说明,而是由系统自动设置的。
四、 写字符函数: int fputc(int ch, file *fp)
ch: 待写入文件的字符的ASCII码
fp: 待写文件的文件指针
返回值: 如写入成功则返回写入的字符, 否则返回EOF
五、读字符串函数:char *fgets(char *str, int num, FILE *fp)
str: 保存从文件读取出来的字符串
num: 表示从文件中读出的字符串不超过 n-1个字符。在读入的最后一个字符后加上串结束标志'/0'
fp: 待读文件的文件指针
返回值: 字符数组的首地址或者NULL(当读到文件末尾或发生错误时返回)
功能描述: 读字符串函数fgets函数的功能是从指定的文件中读一个字符串到字符数组中,如:fgets(str,n,fp)的意义是从fp所指的文件中读出n-1个字符送入字符数组str中。
对fgets函数有两点说明:
1. 在读出n-1个字符之前,如遇到了换行符或EOF,则读出结束。
2. fgets函数也有返回值,其返回值是字符数组的首地址。
六、写字符串函数: int fputs(char *str, file *fp)
str: 待写入文件的字符串
fp: 待写文件的文件指针
返回值: 非负整数(成功),EOF(失败)
功能描述: fputs函数的功能是向指定的文件写入一个字符串
C语言还提供了用于整块数据的读写函数。 可用来读写一组数据,如一个数组元素,一个结构变量的值等。
七、数据块读写函数: int fwrite(void *buf, int size, int count, FILE *fp) | int fread(void *buf, int size, int count, FILE *fp)
buf: 在fread函数中,它表示存放输入数据的首地址。在fwrite函数中,它表示存放输出数据的首地址
size: 表示数据块的字节count: 表示要读写的数据块块数
fp: 表示文件指针
返回: 已读取或已写入的数据块块数
例如:fread(fa,4,5,fp); 其意义是从fp所指的文件中,每次读4个字节(一个实数)送入实数组fa中,连续读5次,即读5个实数到fa中。
八、 格式化读写函数 int fscanf(FILE *fp, char *format,…)| int fprintf(FILE *fp, char *format,…)
fscanf函数,fprintf函数与前面使用的scanf和printf 函数的功能相似,都是格式化读写函数。 两者的区别在于 fscanf 函数和fprintf函数的读写对象不是键盘和显示器,而是磁盘文件。
九、文件的随机读写
前面介绍的对文件的读写方式都是顺序读写, 即读写文件只能从头开始,顺序读写各个数据。 但在实际问题中常要求只读写文件中某一指定的部分。 为了解决这个问题可移动文件内部的位置指针到需要读写的位置,再进行读写,这种读写称为随机读写。 实现随机读写的关键是要按要求移动位置指针,这称为文件的定位。文件定位移动文件内部位置指针的函数主要有两个, 即 rewind 函数和fseek函数。
rewind函数前面已多次使用过,其调用形式为: rewind(文件指针); 它的功能是把文件内部的位置指针移到文件首。 下面主要介绍
fseek函数。
fseek函数用来移动文件内部位置指针,其调用形式为: fseek(文件指针,位移量,起始点); 其中:“文件指针”指向被移动的文件。 “位移量”表示移动的字节数,要求位移量是long型数据,以便在文件长度大于64KB 时不会出错。当用常量表示位移量时,要求加后缀“L”。“起始点”表示从何处开始计算位移量,规定的起始点有三种:文件首,当前位置和文件尾。其表示方法如下:
起始点 表示符号 数字表示
──────────────────────────
文件首 SEEK—SET 0
当前位置 SEEK—CUR 1
文件末尾 SEEK—END 2
[例10.8]在学生文件stu list中读出第二个学生的数据。
#include
struct stu
{
char name[10];
int num;
int age;
char addr[15];
} boy,*qq;
main()
{
FILE *fp;
char ch;
int i=1;
qq=&boy;
if((fp=fopen("stu_list","rb"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
rewind(fp);
fseek(fp,i*sizeof(struct stu),0);
fread(qq,sizeof(struct stu),1,fp);
printf("/n/nname/tnumber age addr/n");
printf("%s/t%5d %7d %s/n",qq->name,qq->num,qq->age,
qq->addr);
}
文件stu_list已由例10.6的程序建立,本程序用随机读出的方法读出第二个学生的数据。程序中定义boy为stu类型变量,qq为指向boy的指针。以读二进制文件方式打开文件,程序第22行移动文件位置指针。其中的i值为1,表示从文件头开始,移动一个stu类型的长度, 然后再读出的数据即为第二个学生的数据。
十、文件检测函数
C语言中常用的文件检测函数有以下几个。
一、文件结束检测函数feof函数调用格式: feof(文件指针);
功能:判断文件是否处于文件结束位置,如文件结束,则返回值为1,否则为0。
二、读写文件出错检测函数ferror函数调用格式: ferror(文件指针);
功能:检查文件在用各种输入输出函数进行读写时是否出错。 如ferror返回值为0表示未出错,否则表示有错。
三、文件出错标志和文件结束标志置0函数clearerr函数调用格式: clearerr(文件指针);
功能:本函数用于清除出错标志和文件结束标志,使它们为0值。
转至:http://blog.csdn.net/zhaozy55555/archive/2008/11/17/3320882.aspx