我们前面学习结构体时,写了通讯录的程序,当通讯录运行起来的时候,可以给通讯录中增加、删除数据,此时数据是存放在内存中,当程序退出的时候,通讯录中的数据自然就不存在了,等下次运行通讯录程序的时候,数据又得重新录入,这样根本没有实用性。我们在想既然是通讯录就应该把信息记录下来,只有我们自己选择删除数据的时候,数据才不复存在。这就涉及到了数据持久化的问题,我们一般数据持久化的方法有,把数据存放在磁盘文件、存放到数据库等方式。使用文件我们可以将数据直接存放在电脑的硬盘上,做到了数据的持久化。
磁盘上的文件是文件。但是在程序设计中,我们一般谈的文件有两种:程序文件、数据文件(从文件功能的角度来分类的,本次讲解的是数据文件。
包括源程序文件(后缀为.c),目标文件(windows环境后缀为.obj),可执行程序(windows环境后缀为.exe)。
文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件。
一个文件要有一个唯一的文件标识,以便用户识别和引用。文件名包含3部分:文件路径+文件名主干+文件后缀例如: c:\code\test.txt为了方便起见,文件标识常被称为文件名。
缓冲文件系统中,关键的概念是“文件类型指针”,简称“文件指针”。每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结体类型是由系统声明的,取名FILE。
每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者不必关心细节。一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便。下面我们可以创建一个FILE*的指针变量:
FILE* pf;//文件指针变量
定义pf是一个指向FILE类型数据的指针变量。可以使pf指向某个文件的文件信息区(是一个结构体变量)。通过该文件信息区中的信息就能够访问该文件。也就是说,通过文件指针变量能够找到与它关联的文件。
文件在读写之前应该先打开文件,在使用结束之后应该关闭文件。在编写程序的时候,在打开文件的同时,都会返回一个FILE*的指针变量指向该文件,也相当于建立了指针和文件的关系。ANSIC 规定使用fopen函数来打开文件,fclose来关闭文件。
//打开文件
FILE*fopen ( constchar*filename, constchar*mode );
//关闭文件
int fclose ( FILE*stream );
我们来简单的介绍一下它们的使用方式:
这是运行之前,下面来看运行之后。
我们会发现在该文件路径下会生成一个test.dat,并且里面还有我们写进去的三个字符。这是fputc的功能,往文件中写入一个字符,接下来让我们看看fgetc,从文件中读取一个字符到屏幕上。
上面是一次只能读写一个字符,接下来我们来看一次能读写一个字符串的。fputs就是一个一次能往文件中写一个字符出的函数,让我们来看看它简单使用。
既然有写那么就会有读,那就让我们来看看读又是什么样的呢?
这就是读取的文件数据的操作了,值得一说的是中间的参数3,实际只会读取俩个字符,还有一个字符是’\0’。
这两个函数与scanf与printf极其相似,使用也是十分相似,一个是从键盘输入和输出到屏幕,一个是输入到文件和从文件中读取,那就让我们具体来看看吧。
fprintf是往文件中输出,函数原型:int fprintf( FILE *stream, const char *format [, argument ]…);第一个参数就是我们要写入的文件指针,后面的参数就与printf一样的,我们需要写入的参数,这里拿结构体来进行举例。
相应的fscanf就是从文件中读取数据,函数原型:int fscanf( FILE *stream, const char *format [, argument ]… )。
这个是以二进制的形式存储数据,也就是当我们从文件中看的时候是看不懂的,函数原型:size_t fwrite( const void *buffer, size_t size, size_t count, FILE *stream );第一个参数是所需要存储数据的地址,第二个是一个元素的大小,第三个参数是元素的个数,第四个就是我们要存入的文件指针,举个例子:
我们会发现除了名字其他的数据我们都看不懂,原因是这种写法是以二级制的形式存储数据的,而字符的二进制形式就是它本身,因此我们只能看懂字符,而看不懂其他的数据。我们再来看看读取的操作,先看函数原型 size_t fread( void *buffer, size_t size, size_t count, FILE *stream );,再来看操作:
关于scanf、printf与fscanf、fprintf在前面已经讲解过了,这里重点讲一下sscanf与sprintf的作用。
它的作用是将一个格式化的数据写入到字符串中,先来看函数原型:int sprintf( char *buffer, const char *format [, argument] … );第一个参数是字符串的地址,后续参数与printf函数一样。
虽然输出看起来像结构体,实际它是存放在字符串中的。
它的作用与sprintf相反,是将字符串中的数据还原为原有的格式,函数原型:int sscanf( const char *buffer, const char *format [, argument ] … );如图:
此时是以结构体的形式输出的。
作用:根据文件指针的位置和偏移量来定位指针,函数原型:int fseek( FILE *stream, long offset, int origin );第一个参数是要操作的文件指针,第二个参数是相对于起始位置的偏移量,第三个是从哪里开始。第三个参数包括:SEEK_CUR(从当前位置开始)、SEEK_END(从文件结束开始)、SEEK_SET从文件开头开始,这有什么用呢?举个例子:
先在文件中存放abcd四个字符。
这时候我们的fseek就派上用场了,我们来看操作。
我们会发现发生了改变,这就是fseek的作用,它可以改变文件指针所指向的位置,其他两个参数也是如此。
返回文件指针相对于起始位置的偏移量。函数原型:long int ftell ( FILE * stream );
可以结合fseek使用,一个是计算相对于其实位置的偏移量,一个是通过偏移量来更改文件指针的指向。
让文件指针的位置回到文件的起始位置。函数原型:void rewind ( FILE * stream );
根据数据的组织形式,数据文件被称为文本文件或者二进制文件。数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件就是文本文件。
一个数据在内存中是怎么存储的呢?
字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。如有整数10000,如果以ASCII码的形式输出到磁盘,则磁盘中占用5个字(每个字符一个字节),而二进制形式输出,则在磁盘上只占4个字节。
牢记:在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束。
而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。
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);
}
ANSIC 标准采用“缓冲文件系统”处理的数据文件的,所谓缓冲文件系统是指系统自动地在内存中为程序中每一个正在使用的文件开辟一块“文件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才一起送到磁盘上。如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的大小根据C编译系统决定的。
本期关于结构体相关的知识就到此结束啦,本期内容不经常使用,如果大家想理解的透彻,建议大家可以多次观看进行理解,再做一些题来进行巩固,在此希望大家都能有所进步喔。
如果觉得本篇文章讲的不错,可以给一个赞来鼓励博主嗷,如果大家有什么看不懂的地方或者发现其中的错误可以在评论区留言或者私信我嗷,那么本期就到此结束,让我们下期再见。