本章讲述的是有关文件的相关内容,本章我们会认识到什么是文件,二进制文件与文本文件,文件的打开和关闭,⽂件的顺序读写和随机读写以及⽂件读取结束的判定和⽂件缓冲区。
1.什么是⽂件
1 程序⽂件
程序⽂件包括源程序⽂件(后缀为.c),⽬标⽂件(windows环境后缀为.obj),可执⾏程序(windows环境后缀为.exe)。
2 数据⽂件
⽂件的内容不⼀定是程序,⽽是程序运⾏时读写的数据,⽐如程序运⾏需要从中读取数据的⽂件,或者输出内容的⽂件。
3.⽂件名
2.⼆进制⽂件和⽂本⽂件
数据在内存中以⼆进制的形式存储,如果不加转换的输出到内存的⽂件中,就是⼆进制⽂件。如果要求在内存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的⽂件就是⽂本⽂件。
例如我们要存储10000,相当于把10000分成‘1’ ‘0’ ‘0’ ‘0’ ‘0’一个一个字符存储,字符‘0’ASCLL码为48,换算成二进制为00110000,而这就是以文本文件存储
3.⽂件的打开和关闭
1.⽂件指针
FILE* pf;//⽂件指针变量
定义pf是⼀个指向FILE类型数据的指针变量。可以使pf指向某个⽂件的⽂件信息区(是⼀个结构体变量)。通过该⽂件信息区中的信息就能够访问该⽂件。也就是说,通过⽂件指针变量能够间接找到与它关联的⽂件。
2.⽂件的打开和关闭
ANSIC 规定使⽤ fopen 函数来打开⽂件, fclose 来关闭⽂件。
//打开⽂件
FILE * fopen ( const char * filename, const char * mode );
//关闭⽂件
int fclose ( FILE * stream );
#include <stdio.h>
int main()
{
FILE* pFile;
//打开⽂件
pFile = fopen("Text.txt", "w");
//⽂件操作
if (pFile != NULL)
{
printf("打开成功");
//关闭⽂件
fclose(pFile);
}
return 0;
}
4.⽂件的顺序读写
功能 |
函数名
|
---|---|
字符输⼊函数
| fgetc |
字符输出函数
| fputc |
⽂本⾏输⼊函数
| fgets |
⽂本⾏输出函数
| fputs |
格式化输⼊函数
| fscanf |
格式化输出函数
| fprintf |
⼆进制输⼊
| fread |
⼆进制输出
|
fwrite
|
格式化输入输出也就是结构体的输入和输出
1.fgetc,fputc
int fgetc ( FILE * stream );
从stream指向的文件中读取一个字符
int fputc ( int character, FILE * stream );
写入一个字符到stream指向的文件中
int main()
{
FILE* pf = fopen("cset.txt", "w");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//读文件
fputc('a', pf);
fclose(pf);
FILE* pf1 = fopen("cset.txt", "r");
if (pf1 == NULL)
{
perror("fopen");
return 1;
}
int ch = fgetc(pf1);
printf("%c\n", ch);
fclose(pf1);
return 0;
}
2.fgets,fputs
int fputs ( const char * str, FILE * stream );
写入一个字符串到stream指向的文件中
char * fgets ( char * str, int num, FILE * stream )
从stream指向的文件中读取num个字符到str指向的空间中
注:当fgets读取文件时,会在末尾加上\0,列如fgest(ch,20,pt),只读取文件中19个字符,剩下一个充当\0
3.fscanf,fprintf
int fscanf ( FILE * stream, const char * format, ... );
从stream指向的文件中读取特定的数据
int fprintf ( FILE * stream, const char * format, ... );
写入特定的数据到stream指向的文件中
4.scanf,sprintf
int sscanf ( const char * s, const char * format, ...);
从s指向的数据(字符串)中读取格式化的数据
int sprintf ( char * str, const char * format, ... );
把格式化的数据写入到str指向的数据(字符串)中
可以发现sprintf把数据转换成了字符,而sscanf把字符内容转换成数据,之后打印
5.fread,fwrite
size_t fwrite ( const void * ptr, size_t size, size_t count, FILE * stream );
从ptr指向的数据,拿取大小为size(单位是字节)数量为count的数据,到stream指向的文件中
size_t fread ( void * ptr, size_t size, size_t count, FILE * stream );
从FILE指向的空间的数据中,拿取大小为size(单位是字节)数量为count的数据,放在ptr所指向的空间中
5.⽂件的随机读写
我们知道文件的读写可以按照顺序来,那么有没有可以让文件随机读写,答案是有的,那我们接下来看看有哪些关于文件随机读写的函数
1.fseek
int fseek ( FILE * stream, long int offset, int origin );
根据⽂件指针的位置和偏移量来定位⽂件指针。
origin有三个参数
SEEK_SET | 文件起始位置 |
SEEK_CUR | 文件末尾 |
SEEK_END | 文件指针的当前位置 |
2.ftell
long int ftell ( FILE * stream );
返回⽂件指针相对于起始位置的偏移量
3.rewind
void rewind ( FILE * stream );
让文件指针返回到初始位置
#include<stdio.h>
int main()
{
FILE* pt = fopen("test.txt", "w");//创建文件
if (pt == NULL)
{
perror(pt);
return;
}
fputs("abcdefg", pt);//输入数据
fclose(pt);
FILE* pf = fopen("test.txt", "r");
char a = fgetc(pf);//拿取第一个数据
printf("%c\n", a);
fseek(pf,3,SEEK_CUR);//那第五个数据,由于之前取过一次,文件指针此时指向b,b+3得到e
a = fgetc(pf);
printf("%c\n", a);
int cle = ftell(pf);
printf("%d\n", cle);
rewind(pf);
a = fgetc(pf);
printf("%c\n", a);
cle = ftell(pf);
printf("%d\n", cle);
fclose(pf);
return 0;
}
6.⽂件读取结束的判定
对于文件读取是否结束,有人可能会用feof函数,其实这是不对的。
feof 的作⽤是:当⽂件读取结束的时候,判断是读取结束的原因是否是:遇到⽂件尾结束。
牢记:在⽂件读取过程中,不能⽤feof函数的返回值直接来判断⽂件的是否结束。
1.⽂本⽂件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets ) 例如:
fgetc 判断是否为 EOF .
fgets 判断返回值是否为 NULL .
2. ⼆进制⽂件的读取结束判断,判断返回值是否⼩于实际要读的个数。 例如:
fread判断返回值是否⼩于实际要读的个数。
7. ⽂件缓冲区
ANSIC 标准采⽤“缓冲⽂件系统” 处理的数据⽂件的,所谓缓冲⽂件系统是指系统⾃动地在内存中为程序中每⼀个正在使⽤的⽂件开辟⼀块“⽂件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才⼀起送到磁盘上。如果从磁盘向计算机读⼊数据,则从磁盘⽂件中读取数据输⼊到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的⼤⼩根据C编译系统决定的
因为有缓冲区的存在,C语⾔在操作⽂件的时候,需要做刷新缓冲区或者在⽂件操作结束的时候关闭⽂ 件。 如果不做,可能导致读写⽂件的问题。