说到文件操作,大家会第一印象想到不就是电脑硬盘中创建文件,写入数据吗,键盘、鼠标就可以搞定,那么接下来我要告诉你的是C语言也可以实现文件操作哦!!!
目录
前言
首先、我们要实现的文件操作,不只是简单的打开和关闭,写入数据,我们也可以从相对应文件中,读取数据,到程序中,使得程序能顺利运行,并对传来的数据进行处理使用!
一、为什么要使用文件操作
我们在前文中,实现了静态、动态通讯录,但是我们也发现了,只要程序关闭,那么添加的通讯录的信息数据,也就一起消失,我们下次打开的时候,通讯录是空的,那么这不太符合我们实际所需。我们需要的是,可以利用、可以存储、可以保存的通讯录,那么我们就必须了解一下,文件操作。
文件操作:使用文件操作我们可以将数据直接存放到电脑的硬盘上,做到数据的持久化
二、文件
磁盘上的文件就是文件。
在程序设计中,我们一般说的文件有两种,分别是程序文件、数据文件(从文件功能的角度进行分类)。
2.1 程序文件
包括源程序文件(后缀为.c),目标文件(windows环境后缀为.obj),可执行程序(windows环境后缀为.exe)。
2.2数据文件
文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件。
由上文对于两种文件的介绍,我们可知,我们需要了解的是数据文件,本文就主要讲解一下,如何处理数据文件,并对该文件进行操作
2.3 文件名
文件名是一个文件的唯一的文件标识,以便于用户识别和引用。
比如:D:\new.txt 这是D盘下的文件名为new的文本文档
三、文件操作
我们之前在写C语言程序的时候,都是通过键盘和屏幕,进行输入和输出数据,我们这一次要将数据输入和读取数据都是于硬盘有关。
3.1 文件打开和关闭
我们首先要了解的是我们之前一直所处理的,输入输出数据都是以终端为对象的,即从终端的键盘输入数据,运行结果显示到显示器上。
如图:
文件操作需要的是,以硬盘中的文件为数据输入输出对象。
如图:
1. 文件指针
在缓冲文件系统中,关键的概念是 “ 文件类型指针 ”,即文件指针
文件信息区:
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型是有系统声明的,取名 FILE.
下图是在VS2013编辑环境提供的stdio.h头文件中有的文件类型的声明:
struct _iobuf {
char *_ptr;
int _cnt;
char *_base;
int _flag;
int _file;
int _charbuf;
int _bufsiz;
char *_tmpfname;
};
typedef struct _iobuf FILE;//就是一个结构体类型的FILE,规定的文件的 所有信息,都是可以存储在上面
不同编译器不同的内容,但是都是可以存放各种文件类型的结构体FILE,正常使用即可,不用深究
每次打开一个文件,系统就会根据文件的情况自动创建一个FILE结构的变量,并填充信息,一般都是通过一个FILE指针来维护这个FILE结构的变量,方便使用
如图:
定义pf是一个指向FILE类型数据的指针变量。可以使pf指向某个文件的文件信息区(是一个结构体变量)。通过该文件信息区中的信息就能够访问该文件。也就是说,通过文件指针变量能够找到与它关联的文件。
2. 文件的打开和关闭
文件在使用前应该先打开,之后再关闭,我们先认识一下C语言给我们提供的文件开关函数
第一个fopen来打开文件,第二个fclose来关闭文件
代码如下:
//打开文件
FILE * fopen ( const char * filename, const char * mode );//返回文件类型
//关闭文件
int fclose ( FILE * stream );//返回整型
打开方式:
1.r表示读 w表示写 a表示追加(即不会覆盖原有数据)
2.带b,标识的是打开二进制文件
3. +号表示读写,如果是r+的话,且没有指定文件,那么会报错,但是w+、a+若没有指定文件,会创建一个指定文件,再输入数据(写文件)
代码演示:
3.2 文件的顺序读写
什么叫做顺序读写呢,意思就是,在读取/写文件的时候,读取一个或者写一个内容进去,光标就会向后移动一位,有顺序的读取和写入
如图:
通过fgetc和fputc函数解释了,文件确实是按照顺序读写的
3.3 函数使用
输入流和输出流介绍:
流的概念:
上面函数使用于所有输入流或文件,这是什么意思?
图示解释如下:
1. fgetc和fputc
上文图示用到这两个函数
代码如下:
//这是fgetc函数,可以理解为得到文件中的一个字符
int main()
{
FILE* pf = fopen("test.txt", "r");
for (int i = 0; i < 10; i++) {
char src = fgetc(pf);
printf("%c", src);
}
fclose(pf);
pf = NULL;
return 0;
}
//这是fputc函数,可以理解为放置文件中一个字符
int main()
{
FILE* pf = fopen("test.txt", "w");
for (int i = 0; i < 26; i++) {
fputc('a' + i, pf);
}
fclose(pf);
pf = NULL;
return 0;
}
实际上,fgetc和fputc函数就是得到或放置一个字符在文件中,返回值为int
2. fgets和fputs
fgets和fputs,可以理解为得到或者放置一个字符串在文件中。
还有一个特点,如图所示:
fgets如果没有读取到内容,返回NULL
代码如下:
//fgets函数
int main()
{
FILE* pf = fopen("test.txt", "r");
char arr[20];
while (fgets(arr, 20, pf) != NULL) {
printf("%s", arr);
}
fclose(pf);
pf = NULL;
return 0;
}
//fputs函数
int main()
{
FILE* pf = fopen("test.txt", "w");
char arr[20];
fputs("hello world!!\n", pf);
fputs("hello why\n", pf);
fputs("hello fzx\n", pf);
fclose(pf);
pf = NULL;
return 0;
}
3. fscanf和fprintf
实际上和scanf、printf差不多的形式,只是多一个参数 FILE*stream
如图:
如果fscanf读取的时候没有数据,那么也不会报错,就认为是没有读取吧
代码如图所示:
struct S {
char name[20];
int age;
double score;
};
//fscanf格式化读取pf流中的数据,赋值给后面的参数,和scanf用法差不多,
//只是多一个FILE*类型,后面于scanf用法一致,改用&就用
int main()
{
FILE* pf = fopen("test.txt", "r+");
//读写都行
//格式化输出
//struct S s = { "why",19,100 };
struct S s = {0};
struct S s1 = { 0 };
fscanf(pf, "%s %d %lf\n", s.name, &(s.age), &(s.score));
printf("%s %d %lf\n", s.name, (s.age), (s.score));
//fscanf(pf, "%s %d %lf\n", s1.name, &(s1.age), &(s1.score));
//printf("%s %d %lf\n", s1.name, (s1.age), (s1.score));
int a = 10;
/*fscanf(pf, "%d", a);
printf("%d\n", a);*/ //这是进行测试如果没有相应的数据读取的时候,不会报错,只是相当于没有这一行代码,不会发生任何改变(对相应a)
fclose(pf);
pf = NULL;
return 0;
}
//这是fprintf函数的使用
int main()
{
FILE* pf = fopen("test.txt", "r+");
//读写都行
//格式化输出
struct S s = { "why",19,100 };
fprintf(pf, "%s %d %lf\n", s.name, (s.age), (s.score));
//printf("%s %d %lf\n", s.name, (s.age), (s.score));
fclose(pf);
pf = NULL;
return 0;
}
1.fscanf 格式化读取pf流中的数据,赋值给后面的参数,和 scanf 用法差不多,只是多一个FILE*类型,后面于 scanf 用法一致,该用&就用
2.fprintf 格式化写入文件中数据,上面代码中,将结构体变量s,数据,写入pf管理的文件中
3.fscanf 如果没有相应的数据读取的时候,不会报错,只是相当于没有这一行代码,不会发生任何改变(对相应a)
4. fread和fwrite
只能接收文件流,二进制文件,可能会有乱码产生
如图分析:
代码演示:
struct S {
char name[20];
int age;
double score;
};
//先写入二进制文件 fwrite
int main()
{
struct S s = { "fzx",18,100 };
FILE* pf = fopen("test.txt", "wb");
//写一个二进制文件
fwrite(&s, sizeof(struct S), 1, pf);
fclose(pf);
pf = NULL;
return 0;
}
//再读取二进制文件 fread
int main()
{
struct S s = { 0 };
FILE* pf = fopen("test.txt", "rb");
//读取文件
fread(&s, sizeof(struct S), 1, pf);
printf("%s %d %lf\n", s.name, s.age, s.score);
return 0;
}
5. 对比一组函数
scanf / fscanf / sscanf
printf / fprintf /sprintf
前面四种大家都认识了,那么sscanf和sprintf是什么呢?
如图所示:
3.4 文件的随机读写
1. fseek
根据文件指针的位置和偏移量来定位文件指针。(改变光标位置)
代码演示:
int main()
{
FILE* pFile;
pFile = fopen("example.txt", "wb");
fputs("This is an apple.", pFile);
fseek(pFile, 9, SEEK_SET);
fputs(" sam", pFile);
fclose(pFile);
return 0;
}
2.ftell
返回文件指针相对于起始位置的偏移量
代码演示:
int main()
{
FILE* pFile;
pFile = fopen("example.txt", "wb");
int num = ftell(pFile);
printf("%d ", num);
return 0;
}
3.rewind
让文件指针的位置回到文件的起始位置
图文演示:
代码演示:
int main()
{
FILE* pf = fopen("test.txt", "w");
for (int i = 0; i < 10; i++) {
fputc('a' + i, pf);
}
int num = ftell(pf);
printf("%d \n", num);
rewind(pf);
printf("%d \n", ftell(pf));
fclose(pf);
pf = NULL;
return 0;
}
3.5 文本文件和二进制文件
1.根据数据的组织形式,数据文件被称为文本文件或者二进制文件。
2.数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。
3.如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储 的文件就是文本文件。
字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。
测试文本文件和二进制文件代码:
int main()
{
int a = 10000;
FILE* pf = fopen("test.txt", "wb");
fwrite(&a, 4, 1, pf);//二进制的形式写到文件中
fclose(pf);
pf = NULL;
return 0;
}
3.6文件读取结束的判定
在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束,而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。
1.文本文件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets )
fgetc 判断是否为 EOF .
fgets 判断返回值是否为 NULL2. 二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。
fread判断返回值是否小于实际要读的个数
图示:
代码演示:
//文本文件
int main(void)
{
int c; // 注意:int,非char,要求处理EOF
FILE* fp = fopen("test.txt", "r");
if(!fp) {
perror("File opening failed");
return EXIT_FAILURE;
}
//fgetc 当读取失败的时候或者遇到文件结束的时候,都会返回EOF
while ((c = fgetc(fp)) != EOF) // 标准C I/O读取文件循环
{
putchar(c);
}//判断是什么原因结束的
if (ferror(fp))
puts("I/O error when reading");
else if (feof(fp))
puts("End of file reached successfully");
fclose(fp);
}
总结
主要讲解文件操作的的一些函数,fopen、fclose、fgets、fwrite等等函数的介绍,以及对于文件的分类、输入和输出流的分析,以及更多文件操作的函数图示分解。
下文我们将使用文件操作,升级通讯录。