第十三章
本章要点
文件的基本概念
文件的基本函数
文件的顺序读写
文件的随机读写
文件简单应用
主要内容
13.1 C文件概述
13.2 文件类型指针
13.3 文件的打开与关闭
13.4 文件的读写
13.5 文件的定位
13.6 出错的检测
13.7 文件输入输出小结
§ 13.1 C文件概述文件的定义所谓文件一般指存储在外部介质 (如磁盘磁带 )上数据的集合,
操作系统是以文件为单位对数据进行管理的,
文件程序数据区输 出 文件缓冲区输 入 文件缓冲区
§ 13.1 C文件概述 (续 )
文件的分类
从用户观点,
特殊文件 (标准输入输出文件或标准设备文件 )
普通文件 (磁盘文件 )
从操作系统的角度看,每一个与主机相连的输入输出设备看作是一个文件。
例,输入文件:终端键盘输出文件:显示屏和打印机文件的分类
按数据的组织形式,
ASCII文件 (文本文件 ):每一个字节放一个 ASCII代码二进制文件,把内存中的数据按其在内存中的存储形式原样输出到磁盘上存放,
例,整数 10000在内存中的存储形式以及分别按 ASCII
码形式和二进制形式输出如下图所示:
§ 13.1 C文件概述 (续 )
§ 13.1 C文件概述 (续 )
文件的分类
ASCII文件和二进制文件的比较:
ASCII文件 便于对字符进行逐个处理,也便于输出字符。但一般占存储空间较多,而且要花费转换时间。
二进制文件 可以节省外存空间和转换时间,但一个字节并不对应一个字符,不能直接输出字符形式。
一般中间结果数据需要暂时保存在外存上,以后又需要输入内存的,常用二进制文件保存。
§ 13.1 C文件概述 (续 )
文件的分类
C语言对文件的处理方法:
缓冲文件系统,系统自动地在内存区为每一个正在使用的文件开辟一个缓冲区。用缓冲文件系统进行的输入输出又称为 高级磁盘输入输出 。
非缓冲文件系统,系统不自动开辟确定大小的缓冲区,而由程序为每个文件设定缓冲区。用非缓冲文件系统进行的输入输出又称为 低级输入输出系统。
§ 13.1 C文件概述 (续 )
说明,
在 UNIX系统下,用缓冲文件系统来处理文本文件,
用非缓冲文件系统来处理二进制文件,
ANSI C 标准只采用缓冲文件系统来处理文本文件和二进制文件,
C语言中对文件的读写都是用库函数来实现,
§ 13.2 文件类型指针
Turbo C在 stdio.h文件中有以下的文件类型声明:
typedef struct
{ shortlevel; / *缓冲区,满,或,空,的程度 */
unsignedflags; / *文件状态标志 */
charfd; / *文件描述符 */
unsignedcharhold; / *如无缓冲区不读取字符 */
shortbsize; / *缓冲区的大小 */
unsignedchar*buffer;/*数据缓冲区的位置 */
unsignedar*curp;/*指针,当前的指向 */
unsignedistemp;/*临时文件,指示器 */
shorttoken;/*用于有效性检查 */} FILE;
在缓冲文件系统中,每个被使用的文件都要在内存中开辟一
FILE类型的区,存放文件的有关信息,
§ 13.2 文件类型指针 (续 )
FILE类型的数组:
FILE f[ 5] ;定义了一个结构体数组 f,它有 5个元素,
可以用来存放 5个文件的信息。
文件型指针变量,
FILE *fp; fp是一个指向 FILE类型结构体的指针变量。可以使 fp指向某一个文件的结构体变量,从而通过该结构体变量中的文件信息能够访问该文件。如果有n个文件,一般应设n个指针变量,使它们分别指向n
个文件,以实现对文件的访问。
§ 13.3 文件的打开与关闭一,文件的打开 (fopen函数 )
函数调用,
FILE *fp;
fp= fopen( 文件名,使用文件方式 );
①需要打开的文件名,也就是准备访问的文件的名字;
②使用文件的方式(,读,还是,写,等);
③让哪一个指针变量指向被打开的文件。
§ 13.3 文件的打开与关闭 (续 )
文件使用方式 含 义
,r” (只读 )为 输入 打开一个 文本 文件
,w” (只写 )为 输出 打开一个 文本 文件
,a” (追加 )向 文本 文件尾增加数据
,rb” (只读 )为 输入 打开一个 二进制 文件
,wb” (只写 )为 输出 打开一个 二进制 文件
"ab,(追加 )向 二进制 文件尾增加数据
"r+,(读写 )为 读 /写 打开一个 文本 文件
"w+” (读写 )为 读 /写建立 一个新的 文本 文件
"a+” (读写 )为读 /写打开一个文本文件
"rb+,(读写 )为读 /写打开一个 二进制 文件
,wb+,(读写 )为读 /写 建立 一个新的 二进制 文件
,ab+” (读写 )为读 /写打开一个 二进制 文件
§ 13.3 文件的打开与关闭 (续 )
二,文件的关闭 (fclose函数 )
函数调用,
fclose( 文件指针 );
函数功能,
使文件指针变量不指向该文件,也就是文件指针变量与文件,脱钩,,此后不能再通过该指针对原来与其相联系的文件进行读写操作返回值,
关闭成功返回值为0;否则返回 EOF(-1)
§ 13.4 文件的读写一,字符输入输出函数 (fputs()和 fgets())
fputs函数函数调用,
fputs ( ch,fp ) ;
函数功能,
将字符( ch的值)输出到 fp所指向的文件中去。
返回值,
如果输出成功,则返回值就是输出的字符;
如果输出失败,则返回一个 EOF.
§ 13.4 文件的读写 (续 )
fgets函数函数调用,
ch= fgets( fp);
函数功能,
从指定的文件读入一个字符,该文件必须是以读或读写方式打开的。
返回值,
读取成功一个字符,赋给ch。
如果遇到文件结束符,返回一个文件结束标志
EOF 。
§ 13.4 文件的读写 (续 )
常见的读取字符操作从一个 文本文件 顺序读入字符并在屏幕上显示出来:
ch = fgetc( fp);
while( ch! =EOF)
{
putchar( ch);
ch = fgetc( fp);

注意,EOF不是可输出字符,因此不能在屏幕上显示。
由于字符的 ASCII码不可能出现-1,因此 EOF定义为
-1是合适的。当读入的字符值等于-1时,表示读入的已不是正常的字符而是文件结束符。
§ 13.4 文件的读写 (续 )
常见的读取字符操作从一个 二进制文件 顺序读入字符:
while(! feof( fp))
{
ch = fgetc( fp);

注意,ANSI C提供一个 feof()函数来判断文件是否真的结束。如果是文件结束,函数 feof( fp)的值为1
(真);否则为0(假)。以上也适用于文本文件的读取。
§ 13.4 文件的读写 (续 )
fputc和 fgetc函数使用举例:
例1 3.1 从键盘输入一些字符,逐个把它们送到磁盘上去,直到输入一个,#,为止。
#include <stdlib.h>
#include <stdio.h>
void main(void)
{ FILE *fp;
char ch,filename[10];
scanf("%s",filename);
if((fp=fopen(filename,"w"))==NULL) {
printf("cannot open file\n");
exit(0); /*终止程序 */}
ch=getchar( ); /*接收执行 scanf语句时最后输入的回车符 */
ch=getchar( ); /* 接收输入的第一个字符 */
while(ch!='#'{
fputc(ch,fp);putchar(ch);
ch=getchar(); }
fclose(fp);
}
运行情况如下:
(输入磁盘文件名)
computer and
computer and c (输出一个字符串)
§ 13.4 文件的读写 (续 )
fputc和 fgetc函数使用举例:
例1 3.2 将一个磁盘文件中的信息复制到另一个磁盘文件中 。
#include <stdlib.h>
#include <stdio.h>
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");
scanf("%s",outfile);
if((in=fopen(infile,"r"))==NULL)
{ printf("cannot open infile\n");
exit(0);}
if((out=fopen(outfile,"w"))==NULL)
{ printf("cannot open outfile\n");
exit(0);}
while(!feof(in))fputc(fgetc(in),out);
fclose(in);
fclose(out);}
运行情况如下:
Enter the infile name
Enter the outfile name:
(输入新复制的磁盘文件名)
程序运行结果是将file1.c文件中的内容复制到
file2.c中去。
§ 13.4 文件的读写 (续 )
fputc和 fgetc函数使用举例:
例1 3.2 的改进:复制一个二进制文件,利用 main
参数,在输入命令行是将两个文件名输入。
#include <stdlib.h>
#include <stdio.h>
main(int argc,char *argv[ ])
{FILE *in,*out;
char ch;
if (argc!=3)
{ printf("You forgot to enter a filename\n");
exit(0); }
if((in=fopen(argv[1],"rb"))==NULL)
{ printf("cannot open infile\n");
exit(0);}
if((out=fopen(argv[2],"wb"))==NULL)
{ printf("cannot open outfile\n");
exit(0);}
while(!feof(in)) fputc(fgetc(in),out);
fclose(in);
fclose(out);}
运行方法:
设经编译连接后得到的可执行文件名为 a.exe,则在 DOS命令工作方式下,可以输入以下的命令行:
C> a file1.c file2.c
file1.c和file2.c,分别输入到argv[1]
和argv[2]中,argv[0]的内容为 a,argc的值等于3 。
§ 13.4 文件的读写 (续 )
二,数据块读写函数 (fread()和 fwrite())
函数调用,
fread (buffer,size,count,fp);
fwrite(buffer,size,count,fp);
参数说明:
buffer,是一个指针。
对 fread 来说,它是读入数据的存放地址。
对 fwrite来说,是要输出数据的地址(均指起始地址)。
size,要读写的字节数。
count,要进行读写多少个 size字节的数据项。
fp,文件型指针。
§ 13.4 文件的读写 (续 )
使用举例:
若 文件以二进制 形式打开:
fread(f,4,2,fp);
此函数从 fp所指向的文件中读入 2个 4个字节的数据,存储到数组 f中。
§ 13.4 文件的读写 (续 )
使用举例:
若有如下结构类型:
struct student_type
{char name[10];
int num;
int age;
char addr[30];}stud[40];
可以用 fread和 fwrite来进行数据的操作:
for(i=0;i<40;i++)
fread(&stud[ i],sizeof(struct student-type),1,fp);
for(i=0;i<40,i++)
fwrite(&stud[ i],sizeof(struct student-type),1,fp);
§ 13.4 文件的读写 (续 )
使用举例:
例1 3.3 从键盘输入4个学生的有关数据,然后把它们转存到磁盘文件上去。
#include <stdio.h>
#define SIZE 4
struct student_type
{ char name[10];
int num;
int age;
char addr[15];
}stud[SIZE]; /*定义结构 */
§ 13.4 文件的读写 (续 )void save( ){FILE *fp;
int i;
if((fp=fopen("stu-list","wb"))==NULL)
{ printf("cannot 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”);/*出错处理 */
fclose(fp); } /*关闭文件 */
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( );}/*调用 save()保存学生信息 */
运行情况如下:
输入4个学生的姓名、学号、年龄和地址:
Zhang 1001 19 room -
Fun 1002 20 room -
Tan 1003 21 room -
Ling 1004 21 room -
验证 在磁盘文件,stu -list,中是否已存在此数据,
用以下程序从,stu -list,文件中读入数据,然后在屏幕上输出。
§ 13.4 文件的读写 (续 )
#include <stdio.h>
#define SIZE 4
struct student_type
{ char name[10];
int num;
int age;
char addr[15];
}stud[SIZE];
main( )
{ int i;
FILE*fp;
fp=fopen("stu-list","rb");
for(i=0;i<SIZE;i++)
{fread(&stud[i],sizeof(struct student_type),1,fp);
printf("%\-10s %4d %4d %\-15s\n",stud[i].name,
stud[i].num,stud[i],age,stud[i].addr); }
fclose (fp);}
屏幕上显示出以下信息:
Zhang 1001 19 room -101
Fun 1002 20 room -102
Tan 1003 21 room -103
Ling 1004 21 room -104
如果已有的数据已经以 二进制形式 存储在一个磁盘文件
,stu -dat,中,要求从其中读入数据并输出到
,stu -list,文件中,可以编写一个 load函数,
从磁盘文件中读二进制数据 。
§ 13.4 文件的读写 (续 )
void load( )
{FILE *fp;int i;
if((fp=fopen("stu-dat","rb"))==NULL)
{ printf("cannot open infile\n");
return;}
for(i=0;i<SIZE;i++)
if(fread(&stud[i],sizeof(struct student_type),1,fp)!=1)
{if(feof(fp)) {fclose(fp); return;}
printf("file read error\n");}
fclose (fp); }
§ 13.4 文件的读写 (续 )
三,格式化读写函数 (fprintf()和 fscanf())
函数调用,
fprintf ( 文件指针,格式字符串,输出表列);
fscanf ( 文件指针,格式字符串,输入表列);
函数功能,
从磁盘文件中读入或输出字符。
例:
fprintf(fp,”%d,%6.2f”,i,t);
Fscanf (fp,”%d,%f”,&i,&t);
注意:
用 fprintf和 fscanf函数对磁盘文件读写,使用方便,容易理解,
但由于在输入时要将 ASCII码转换为二进制形式,在输出时又要将二进制形式转换成字符,花费时间比较多。因此,在 内存与磁盘频繁交换数据 的情况下,最好不用 fprintf和 fscanf函数,而用 fread和 fwrite函数。
§ 13.4 文件的读写 (续 )
三,其他读写函数
putw()和 getw()
函数调用,
putw(int i,FILE * fp);
int i = getw(FILE * fp);
函数功能,
对磁盘文件中读写一个字(整数)。
例:
putw(10,fp);
i = getw(fp);
putw函数定义如下:
putw(int I,FILE *fp)
{
char s;
s=&I;
putc(s[0],fp);
putc(s[1],fp);
return i;
}
gutw函数定义如下:
gutw(FILE *fp)
s=char *&i;
s[0] = getc(fp);
s[1] = getc(fp);
§ 13.4 文件的读写 (续 )
用户自定义读取其他类型数据的函数向磁盘文件写一个实数(用二进制方式)的函数 putfloat,
putfloat(float num,FILE *fp)
{
char s;
int count;
s = (char*)&num;
for(count = 0;count < 4;count++)
putc(s[count],fp);
}
§ 13.4 文件的读写 (续 )
fgets函数函数作用:
从指定文件读入一个字符串。
函数调用:
fgets(str,n,fp);
从 fp指向的文件输入 n-1个字符,在最后加一个 ’ \0’
返回值:
str的首地址
§ 13.4 文件的读写 (续 )
fputs函数函数作用:
向指定的文件输出一个字符串。
函数调用:
fgets(“china”,fp);
第一个参数可以是字符串常量、字符数组名或字符型指针。字符串末尾的 ′ \0 ′ 不输出。
返回值:
输入成功,返回值为 0;
输入失败,返回 EOF.
§ 13.5 文件的定位
rewind函数函数作用:
使位置指针重新返回文件的开头,无返回值。
应用举例,
例1 3.4 有一个磁盘文件,第一次将它的内容显示在屏幕上,第二次把它复制到另一文件上。
#include<stdio.h>
main()
{ FILE *fp1,*fp2;
fp1=fopen("file1.c","r");
fp2=fopen("file2.c","w");
while(!feof(fp1)) putchar(getc(fp1));
rewind(fp1);
while(!feof(fp1))
putc(getc(fp1),fp2);
fclose(fp1);fclose(fp2);
}
§ 13.5 文件的定位顺序读写和随机读写顺序读写:
位置指针按字节位置顺序移动。
随机读写,
读写完上一个字符(字节)后,并不一定要读写其后续的字符(字节),而可以读些文件中任意位置上所需要的字符(字节)。
§ 13.5 文件的定位
fseek函数(一般用于二进制文件)
函数功能:
改变文件的位置指针函数调用形式,
fseek(文件类型指针,位移量,起始点 )
起始点:文件开头 SEEK_SET 0
文件当前位置 SEEK_CUR 1
文件末尾 SEEK_END 2
位移量:以起始点为基点,向前移动的字节数。一般要求为 long型
§ 13.5 文件的定位
fseek函数应用举例
fseek(fp,100L,0);
将位置指针移到离文件头 100个字节处
fseek(fp,50L,1);
将位置指针移到离当前位置 50个字节处
fseek(fp,50L,2);
将位置指针从文件末尾处向后退10个字节
§ 13.5 文件的定位例1 3.5 在磁盘文件上存有10个学生的数据。要求将第1、3、5、7、9个学生数据输入计算机,并在屏幕上显示出来。
#include <stdlib.h>
#include<stdio.h>
struct student_type
{ char name[10];
int num;
int age;
char sex;
}stud[10]; main()
{ int i;
FILE *fp;
if((fp=fopen("stud-dat","rb"))==NULL)
{printf("can not open file\n");
exit(0);}
for(i=0;i<10;i+=2)
{fseek(fp,i*sizeof(struct student_type),0);
fread(&stud[i],sizeof(struct student_type),1,fp);
printf(,%s %d %d %c\n”,stud[i].name,
stud[i].num,stud[i].age,stud[i].sex);}
fclose(fp)}
§ 13.5 文件的定位
ftell函数函数作用:
得到流式文件中的当前位置,用相对于文件开头的位移量来表示。
返回值:
返回当前位置,出错时返回 -1L。
应用举例:
i = ftell(fp);
if(i==-1L) printf(“error\n”);
§ 13.6 出错的检测
ferror函数调用形式:
ferror(fp);
返回值:
返回 0,表示未出错;返回非 0,表示出错。
在调用一个输入输出函数后立即检查 ferror函数的值,
否则信息会丢失。在执行 fopen函数时,ferror函数的初始值自动置为0。
§ 13.6 出错的检测
clearerr函数调用形式:
clearerr(fp);
函数作用:
使文件错误标志和文件结束标志置为 0。
只要出现错误标志,就一直保留,直到对同一文件调用 clearerr函数或 rewind函数,或任何其他一个输入输出函数。
§ 13.7 文件输入输出小结分类 函数名 功能打开文件 fopen() 打开文件关闭文件 fclose() 关闭文件文件定位 fseek() 改变文件位置指针的位置
Rewind() 使文件位置指针重新至于文件开头
Ftell() 返回文件位置指针的当前值文件状态 feof() 若到文件末尾,函数值为真
Ferror() 若对文件操作出错,函数值为真
Clearerr() 使 ferror和 feof()函数值置零
§ 13.7 文件输入输出小结分类 函数名 功能文件读写 fgetc(),getc()从指定文件取得一个字符
fputc(),putc()把字符输出到指定文件
fgets()从指定文件读取字符串
fputs()把字符串输出到指定文件
getw()从指定文件读取一个字( int型)
putw()把一个字输出到指定文件
fread()从指定文件中读取数据项
fwrite()把数据项写到指定文件中
fscanf()从指定文件按格式输入数据
fprintf()按指定格式将数据写到指定文件中