第 13章 文件
教学目与要求:
明确文件、文件指针的概念,掌握文件的打开与关闭,
文件的顺序读写,文件的定位与随机读写的方法。了解文件操作的出错检测方法。
教学内容:
C文件概述;文件类型指针;文件的打开与关闭;文件的读写;文件的定位;出错的检测;文件的输入输出小结。
教学重点与难点:
文件的打开与关闭,顺序读写方法,文件的定位与随机读写方法。
13.1 C文件概述
文件:存储在外部介质上数据的集合,是操作系统数据管理的单位
文件分类
按存储介质:
普通文件:存储介质文件(磁盘、磁带等)
设备文件,非存储介质(键盘、显示器、打印机等)
按数据的组织形式:
文本文件,ASCII文件,每个字节存放一个字符的 ASCII码
二进制文件,数据按其在内存中的存储形式原样存放
按文件的逻辑结构:
记录文件:由具有一定结构的记录组成(定长和不定长)
流式文件,由一个个字符(字节)数据顺序组成使用数据文件的目的
1、数据文件的改动不引起程序的改动 —— 程序与数据分离
2、不同程序可以访问同一数据文件中的数据 —— 数据共享
3、能 长期保存 程序运行的中间数据或结果数据如 int型数 10000
0010011100010000
内存存储形式
0010011100010000
二进制形式
0011000100110000001100000011000000110000
ASCII形式文本文件特点,
存储量大、速度慢、便于对字符操作二进制文件特点,
存储量小、速度快、便于存放中间结果
文件处理方法
缓冲文件系统,高级 文件系统,系统自动为正在使用的文件开辟内存缓冲区
非缓冲文件系统,低级文件系统,由用户在程序中为每个文件设定缓冲区磁盘文件输出文件缓冲区输入文件缓冲区程序数据区
a
缓冲文件系统:
缓冲区指令区程序用户数据区 磁盘非缓冲文件系统:
13.2 文件类型指针
文件结构体 FILE
缓冲文件系统为每个正使用的文件在内存开辟文件信息区
文件信息用系统定义的名为 FILE的结构体描述
FILE定义在 stdio.h中
typedef struct
{ int _fd; //文件号
int _cleft; //缓冲区中剩下的字符数
int _mode; //文件操作方式
char *_next; //文件当前读写位置
char *_buff; //文件缓冲区位置
}FILE;
FILE f[5];
文件类型指针
指针变量说明,FILE *fp;
用法:
文件打开 时,系统 自动 建立文件结构体,并把指向它的指针返回来,程序通过这个指针获得 文件信息,访问文件
文件关闭 后,它的文件结构体被释放文件名文件使用方式文件类型指针
C程序 操作系统磁盘
13,3文件的打开与关闭
C文件操作用 库函数 实现,包含在 stdio.h
文件使用方式,打开文件 -->文件读 /写 -->关闭文件
系统 自动 打开和关闭三个 标准文件,
标准输入 ------键盘 stdin
标准输出 ------显示器 stdout
标准出错输出 -----显示器 stderr
13.3.1打开文件 fopen
函数原型,FILE *fopen(文件名,使用文件方式 )
功能:按指定方式打开文件
返值:正常打开,为指向文件结构体的指针;打开失败,为 NULL
要打开的文件名使用文件方式例 文件打开与测试
FILE *fp;
fp=fopen(“aa.c”,“w”);
if(fp==NULL)
{ printf(“File open error!\n”);
exit(0);
}
例 FILE *fp;
fp= fopen (“c:\\fengyi\\bkc\\test.dat”,”r”);
例 FILE *fp;
char *filename=“c:\\fengyi\\bkc\\test.dat”
fp= fopen(filename,”r”);
“r+/rb+” (读写 )
“a/ab” (追加 )
“w/wb” (只写 )
“r/rb” (只读 )
“w+/wb+” (读写 )
“a+/ab+” (读写 )
为 输入 打开一个文本 /二进制文件为 输出 打开或建立一个文本 /二进制文件为读 /写打开一个文本 /二进制文件为读 /写建立一个文本 /二进制文件为读 /写打开或建立一个文本 /二进制文件向文本 /二进制文件尾 追加 数据文件使用方式 含义
13.3.2文件关闭 fclose
作用,使文件指针变量与文件“脱钩”,释放文件结构体和文件指针
函数原型,int fclose(FILE *fp)
功能:关闭 fp指向的文件
返值:正常关闭为 0;出错时,非 0( EOF,值为 -1)文件打开时返回的文件类型指针磁盘文件输出文件缓冲区输入文件缓冲区程序数据区
a
缓冲文件系统:
fclose
不关闭文件可能会 丢失数据
13.4 文件的读写
13.4.1字符 I/O:fputc与 fgetc(putc和 getc)
1.fputc
函数原型,int fputc(char c,FILE *fp)
功能:把一字符 c写入 fp指向的文件中
返值:正常,返回 c;出错,为 EOF( -1)
2.fgetc
函数原型,int fgetc(FILE *fp)
功能:从 fp指向的文件中读取 一字节代码
返值:正常,返回读到的代码值 ;读到文件尾或出错,为 EOF
文件 I/O与终端 I/O
#define putc(ch,fp) fputc(ch,fp)
#define getc(fp) fgetc(fp)
#define putchar( c ) fputc(c,stdout)
#define getchar( ) fgetc(stdin)
判断 二进制 文件是否结束
while(!feof(fp))
{ c=fgetc(fp);
……..
}
例 文件拷贝例 从键盘输入字符,逐个存到磁盘文件中,直到输入‘ #“为止
#include <stdio.h>
void main()
{ FILE *fp;
char ch,*filename=“out.txt”;
if((fp=fopen(filename,"w"))==NULL)
{ printf("cannot open file\n");
exit(0);
}
printf("Please input string:");
ch=getchar();
while(ch!='#')
{ fputc(ch,fp);
putchar(ch);
ch=getchar();
}
fclose(fp);
}
例 读文本文件内容,
并显示
#include <stdio.h>
void main()
{ FILE *fp;
char ch,*filename=“out.txt”;
if((fp=fopen(filename,”r"))==NULL)
{ printf("cannot open file\n");
exit(0);
}
while((ch=fgetc(fp))!=EOF)
putchar(ch);
fclose(fp);
}
判断文本文件是否结束? feof
函数原型,int feof(FILE *fp)
功能:判断文件是否结束
返值:文件结束,返回真(非 0);文件未结束,返回 0
#include <stdio.h>
void main()
{ FILE *in,*out;
char ch,infile[10],outfile[10];
scanf("%s",infile);
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);
}
13.4.2 数据块 I/O:fread与 fwrite
调用形式:
fread(buffer,size,count,fp)
fwrite(buffer,size,count,fp)
功能:读 /写数据块
返值:成功,返回读 /写的块数;出错或文件尾,返回 0
说明:
buffer,指向要输入 /输出数据块的首地址的指针
size,每个要读 /写的数据块的大小(字节数)
count,要读 /写的数据块的个数
fp,要读 /写的文件指针
fread与 fwrite 一般用于 二进制文件 的输入 /输出例 float f[2];
FILE *fp;
fp=fopen(“aa.dat”,“rb”);
fread(f,4,2,fp);
for(i=0;i<2;i++)
fread(&f[i],4,1,fp);
例 struct student
{ int num;
char name[20];
char sex;
int age;
float score[3];
}stud[10];
for(i=0;i<10;i++)
fread(&stud[i],sizeof(struct student),1,fp);
例 从键盘输入 4个学生数据,把他们转存到磁盘文件中去
#include <stdio.h>
#define SIZE 4
struct student_type
{ char name[10];
int num;
int age;
char addr[15];
}stud[SIZE];
void main()
{
void save();
void display();
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();
display();
}
void save()
{ FILE *fp;
int i;
if((fp=fopen("d:\\fengyi\\exe\\stu_dat","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);
}
void display()
{ FILE *fp;
int i;
if((fp=fopen("d:\\fengyi\\exe\\stu_dat","rb"))==NULL)
{ printf("cannot open file\n");
return;
}
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);
}
13.5文件的定位
几个概念
文件位置指针 -----指向当前读写位置的指针
读写方式
顺序读写:位置指针按字节位置顺序移动,叫 ~
随机读写:位置指针按需要移动到任意位置,叫 ~
13.5.1 rewind函数
函数原型,void rewind(FILE *fp)
功能:重置文件位置指针到文件开头
返值:无例 对一个磁盘文件进行显示和复制两次操作
#include <stdio.h>
void main()
{ FILE *fp1,*fp2;
fp1=fopen("d:\\fengyi\\bkc\\ch12_4.c","r");
fp2=fopen("d:\\fengyi\\bkc\\ch12_41.c","w");
while(!feof(fp1)) putchar(getc(fp1));
rewind(fp1);
while(!feof(fp1)) putc(getc(fp1),fp2);
fclose(fp1);
fclose(fp2);
}
13.5.2 fseek函数
函数调用,fseek(文件类型指针,位移量,起始点 )
功能:改变文件位置指针的位置
返值:成功,返回 0;失败,返回非 0值文件指针位移量(以起始点为基点,移动的字节数 )
>0 向后移动
<0 向前移动起始点文件开始 SEEK_SET 0
文件当前位置 SEEK_CUR 1
文件末尾 SEEK_END 2
例 fseek(fp,100L,0);
fseek(fp,50L,1);
fseek(fp,-10L,2);
13.5.3 ftell函数
函数原型,long ftell(FILE *fp)
功能:返回位置指针当前位置 (用相对文件开头的位移量表示 )
返值:成功,返回当前位置指针位置;失败,返回 -1L,
例 磁盘文件上有 3个学生数据,要求读入 1,3学生数据并显示
#incldue <stdio.h>
#include <stdlib.h>
void main()
{ int i;
FILE *fp;
if((fp=fopen("studat","rb"))==NULL)
{ printf("can't open file\n");exit(0); }
for(i=0;i<3;i+=2)
{ fseek(fp,i*sizeof(struct student_type),0);
fread(&stud[i],sizeof(struct student_type),1,fp);
printf("%s %d %d %s\n",
stud[i].name,stud[i].num,stud[i].age,stud[i].addr);
}
fclose(fp);
}
#include <stdio.h>
struct student_type
{ int num;
char name[10];
int age;
char addr[15];
}stud[3];
#include <stdio.h>
void main()
{ FILE *fp;
char filename[80];
long length;
gets(filename);
fp=fopen(filename,"rb");
if(fp==NU L)
printf("file not found!\n");
else
{ fseek(fp,0L,SEEK_END);
length=ftell(fp);
printf("Length of File is %1d bytes\n",length);
fclose(fp);
}
}
例 求文件长度 (ch12_101.c)
13.6 出错的检测
13.6.1 ferror函数
函数原型,int ferror(FILE *fp)
功能:测试文件是否出现错误
返值:未出错,0;出错,非 0
说明
每次调用文件输入输出函数,均产生一个新的 ferror函数值,
所以应及时测试
fopen打开文件时,ferror函数初值自动置为 0
13.6.2 clearerr函数
函数原型,void clearerr(FILE *fp)
功能:使文件错误标志置为 0
返值:无
说明:出错后,错误标志一直保留,直到对同一文件调 clearerr(fp)或 rewind或任何其它一个输入输出函数例 ferror()与 clearerr()举例
#include <stdio.h>
int main(void)
{ FILE *stream;
stream = fopen("DUMMY.FIL","w");
getc(stream);
if (ferror(stream))
{ printf("Error reading from DUMMY.FIL\n");
clearerr(stream);
}
if(!ferror(stream))
printf("Error indicator cleared!");
fclose(stream);
return 0;
}
作业
13.4
13.5
13.6