第 9章 文件
§ 9.1 C文件概述
文件:存储在外部介质上数据的集合,是操作系统数据管理的单位使用数据文件的目的
1、数据文件的改动不引起程序的改动 ——程序与数据分离
2、不同程序可以访问同一数据文件中的数据 ——数据共享
3、能 长期保存 程序运行的中间数据或结果数据
文件分类
按文件的逻辑结构:
记录文件:由具有一定结构的记录组成(定长和不定长)
流式文件,由一个个字符(字节)数据顺序组成
按存储介质:
普通文件:存储介质文件(磁盘、磁带等)
设备文件,非存储介质(键盘、显示器、打印机等)
按数据的组织形式:
文本文件,ASCII文件,每个字节存放一个字符的 ASCII码
二进制文件,数据按其在内存中的存储形式原样存放如 int型数 10000
0010011100010000
内存存储形式
0010011100010000
二进制形式
0011000100110000001100000011000000110000
ASCII形式文本文件特点,
存储量大、速度慢、便于对字符操作二进制文件特点,
存储量小、速度快、便于存放中间结果
文件处理方法
缓冲文件系统,高级 文件系统,系统自动为正在使用的文件开辟内存缓冲区
非缓冲文件系统,低级文件系统,由用户在程序中为每个文件设定缓冲区磁盘文件输出文件缓冲区输入文件缓冲区程序数据区
a
缓冲文件系统:
缓冲区指令区程序用户数据区 磁盘非缓冲文件系统:
§ 9.2 文件类型指针
文件结构体 FILE
缓冲文件系统为每个正使用的文件在内存开辟文件信息区
文件信息用系统定义的名为 FILE的结构体描述
FILE定义在 stdio.h中
typedef struct
{ int _fd; //文件号
int _cleft; //缓冲区中剩下的字符数
int _mode; //文件操作方式
char *_next; //文件当前读写位置
char *_buff; //文件缓冲区位置
}FILE;
文件类型指针
指针变量说明,FILE *fp;
用法:
文件打开 时,系统 自动 建立文件结构体,并把指向它的指针返回来,程序通过这个指针获得 文件信息,访问文件
文件关闭 后,它的文件结构体被释放文件结构体文件类型指针磁盘
fp
§ 9.3 文件的打开与关闭
C文件操作用库函数实现,库函数包含于 stdio.h
★ 三个标准文件:
标准输入 ------键盘 stdin
标准输出 ------显示器 stdout
标准出错输出 -----显示器 stderr
对这三个 标准文件 系统是 自动 打开和关闭(不需要语句来操作)
★ 对文件的读写应该按指定的操作方式打开打开文件使用方式,打开文件 -->文件读 /写 -->关闭文件
★ 打开文件 fopen
函数原型,FILE *fopen(char *name,char *mode)
功能:按指定方式打开指定的文件
返值:正常打开,为指向文件结构体的指针;打开失败,为 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+” (读写 )
为 输入 打开一个文本 /二进制文件为 输出 打开或建立一个文本 /二进制文件为读 /写打开一个文本 /二进制文件为读 /写建立一个文本 /二进制文件为读 /写打开或建立一个文本 /二进制文件向文本 /二进制文件尾 追加 数据文件使用方式 含义
文件关闭 fclose
作用,使文件指针变量与文件?脱钩?,释放文件结构体和文件指针
函数原型,int fclose(FILE *fp)
功能:关闭 fp指向的文件
返值:正常关闭为 0;出错时,非 0文件打开时返回的文件类型指针磁盘文件输出文件缓冲区输入文件缓冲区程序数据区
a
缓冲文件系统:
fclose
不关闭文件可能会 丢失数据
§ 9.4 文件的读写
字符 I/O:fputc与 fgetc
fputc
函数原型,int fputc(int c,FILE *fp)
功能:把一字节代码 c写入 fp指向的文件中
返值:正常,返回 c;出错,为 EOF
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)
例 9.1从键盘输入字符,逐个存到磁盘文件中,直到输入‘ #“为止
#include <stdio.h>
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>
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);
} 判断文本文件是否结束例 13.1
判断 二进制 文件是否结束
while(!feof(fp))
{ c=fgetc(fp);
……..
}
例 9.2 文件拷贝
feof
函数原型,int feof(FILE *fp)
功能:判断文件是否结束
返值:文件结束,返回真(非 0);文件未结束,返回 0
#include <stdio.h>
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);
}
数据块 I/O:fread与 fwrite
函数原型:
size_t fread(void *buffer,size_t size,size_t count,FILE *fp)
size_t fwrite(void *buffer,size_t size,size_t count,FILE *fp)
功能:读 /写数据块
返值:成功,返回读 /写的块数;出错或文件尾,返回 0
说明:
typedef unsigned size_t;
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);
例 9.3 从键盘输入 4个学生数据,把他们转存到磁盘文件中去
#include <stdio.h>
#define SIZE 2
struct student_type
{ char name[10];
int num;
int age;
char addr[15];
}stud[SIZE];
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();
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);
}
格式化 I/O:fprintf与 fscanf
函数原型:
int fprintf(FILE *fp,const char *format[,argument,…])
int fscanf(FILE *fp,const char *format[,address,…])
功能:按格式对文件进行 I/O操作
返值:成功,返回 I/O的个数 ;出错或文件尾,返回 EOF
例 fprintf(fp,“%d,%6.2f”,i,t); //将 i和 t按 %d,%6.2f格式输出到 fp文件
fscanf(fp,“%d,%f”,&i,&t); //若文件中有 3,4.5,则将 3送入 i,4.5送入 t
例 从键盘按格式输入数据存到磁盘文件中去
#include <stdio.h>
main()
{ char s[80],c[80];
int a,b;
FILE *fp;
if((fp=fopen("test","w"))==NULL)
{ puts("can't open file"); exit() ; }
fscanf(stdin,"%s%d",s,&a);/*read from keaboard*/
fprintf(fp,"%s %d",s,a);/*write to file*/
fclose(fp);
if((fp=fopen("test","r"))==NULL)
{ puts("can't open file"); exit(); }
fscanf(fp,"%s%d",c,&b);/*read from file*/
fprintf(stdout,"%s %d",c,b);/*print to screen*/
fclose(fp);
}
字符串 I/O,fgets与 fputs
函数原型,char *fgets(char *s,int n,FILE *fp)
int fputs(char *s,FILE *fp)
功能:从 fp指向的文件读 /写一个字符串
返值:
fgets正常时返回读取字符串的首地址;出错或文件尾,返回
NULL
fputs正常时返回写入的最后一个字符;出错为 EOF
例 从键盘读入字符串存入文件,再从文件读回显示
fgets从 fp所指文件读 n-1个字符送入 s指向的内存区,
并在最后加一个 ‘ \0’
(若读入 n-1个字符前遇换行符或文件尾( EOF) 即结束 )
fputs把 s指向的字符串写入 fp指向的文件
#include<stdio.h>
main()
{ FILE *fp;
char string[81];
if((fp=fopen("file.txt","w"))==NULL)
{ printf("cann't open file");exit(0); }
while(strlen(gets(string))>0)
{ fputs(string,fp);
fputs("\n",fp);
}
fclose(fp);
if((fp=fopen("file.txt","r"))==NULL)
{ printf("cann't open file");exit(0); }
while(fgets(string,81,fp)!=NULL)
fputs(string,stdout);
fclose(fp);
}
§ 9.5 文件的定位
几个概念
文件位置指针 -----指向当前读写位置的指针
读写方式
顺序读写:位置指针按字节位置顺序移动,叫 ~
随机读写:位置指针按需要移动到任意位置,叫 ~
rewind函数
函数原型,void rewind(FILE *fp)
功能:重置文件位置指针到文件开头
返值:无例 对一个磁盘文件进行显示和复制两次操作
#include <stdio.h>
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);
}
fseek函数
函数原型,int fseek(FILE *fp,long offset,int whence)
功能:改变文件位置指针的位置
返值:成功,返回 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);
ftell函数
函数原型,long ftell(FILE *fp)
功能:返回位置指针当前位置 (用相对文件开头的位移量表示 )
返值:成功,返回当前位置指针位置;失败,返回 -1L,
例 磁盘文件上有 3个学生数据,要求读入第 1,3学生数据并显示
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"
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)
§ 9.6 出错的检测
ferror函数
函数原型,int ferror(FILE *fp)
功能:测试文件是否出现错误
返值:未出错,0;出错,非 0
说明
每次调用文件输入输出函数,均产生一个新的 ferror函数值,
所以应及时测试
fopen打开文件时,ferror函数初值自动置为 0
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;
}