目录
前言
本文主要介绍文件的相关定义,文件的打开和关闭以及读写函数,最后实现通讯录的文件版。
什么是文件
前面我模拟实现的通讯录其所存储的数据是在内存中存储的,当我关闭程序后,我所输入的联系人信息就被销毁了,当我再次打开通讯录,里面没有任何信息,那怎样永久保存信息呢?这里就需要保存到文件了。
文件就是磁盘上的文件,从文件功能角度来说一般分为两种:程序文件和数据文件。
程序文件:源文件程序(.c),目标文件(.o/.obj),可执行程序(.exe)。
数据文件:文件的内容是程序运行时读写的数据,比如程序要读取的数据来源于文件。
文件是有文件名的,怎么查看文件名,右击文件查看属性即可看到。
文件的打开和关闭
文件指针
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型是有系统声明的,取名FILE。不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异。每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者不必关心细节。一般都是通过一个FILE的指针来维护这个FILE结构的变量,也就是说通过FILE指针能够找到和他相关联的文件。
文件的打开和关闭
打开(fopen)
通过函数fopen函数可以打开文件,那先来认识函数:
fopen函数有两个参数,第一个参数filename是文件名,第二个参数mode是文件使用方式,返回类型是文件指针。关于第一个参数,如果文件和源文件在同一目录下,只输入文件名即可,如果不在,就需要输入绝对路径加文件名;第二个参数的取值是有限制的,具体见下表:
文件使用方式 | 含义 | 如果指定文件不存在 |
---|---|---|
“r”(只读) | 为了输入数据,打开一个已经存在的文本文件 | 出错 |
“w”(只写) | 为了输出数据,打开一个文本文件 | 建立一个新的文件 |
“a”(追加) | 向文本文件尾添加数据 | 建立一个新的文件 |
“rb”(只读) | 为了输入数据,打开一个二进制文件 | 出错 |
“wb”(只写) | 为了输出数据,打开一个二进制文件 | 建立一个新的文件 |
“ab”(追加) | 向一个二进制文件尾添加数据 | 出错 |
“r+”(读写) | 为了读和写,打开一个文本文件 | 出错 |
“w+”(读写) | 为了读和写,创建一个文本文件 | 创建一个新文件 |
“a+”(读写) | 打开一个文件,在文件尾进行读写 | 创建一个新文件 |
“rb+”(读写) | 为了读和写打开一个二进制文件 | 出错 |
“wb+”(读写) | 为了读和写,新建一个新的二进制文件 | 创建一个新文件 |
“ab+”(读写) | 打开一个二进制文件,在文件尾进行读和写 | 创建一个新文件 |
关于文本文件和二进制文件:
数据文件分为两种文本文件和二进制文件。二进制文件:数据在内存中以二进制的形式存储,如果不加转换的输出到外存,形成二进制文件;文本文件:如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件称为文本文件。
关闭(fclose)
通过fclose函数来关闭文件,首先认识函数:
fclose函数只有一个参数stream,就是文件指针,指向打开的那个文件,返回值为整形。
认识完打开关闭函数,通过实例来感受一下:
#include <stdio.h>
int main()
{
//打开文件
FILE* pa = fopen("test.txt", "w");
if (pa == NULL)
{
perror("fopen");//打印错误信息和strerror作用相同
return 1;
}
//文件输出
//关闭文件
fclose(pa);
pa = NULL;
return 0;
}
值得注意的几点:
(1)打开文件时,打开方式即使是一个字母,也要用双引号;
(2)打开文件返回的文件指针不能随意改动,否则会导致文件关闭失败;
(3)和动态内存函数类似,打开文件函数和关闭文件函数也要成对使用,否则可能会导致数据丢失。
(4)fopen函数打开文件返回的文件指针不能直接使用,要进行判断,fopen函数打开成功返回文件指针,打开失败返回NULL;fclose函数关闭失败返回EOF,关闭成功返回0。
文件的读写
文件的顺序读写
文件的顺序读写有以下函数:
功能 | 函数名 | 适用于 |
---|---|---|
字符输入函数 | fgetc | 所有输入流 |
字符输出函数 | fputc | 所有输入流 |
文本行输入函数 | fgets | 所有输入流 |
文本行输出函数 | fputs | 所有输入流 |
格式化输入函数 | fscanf | 所有输入流 |
格式化输出函数 | fprintf | 所有输入流 |
二进制输入 | fread | 文件 |
二进制输出 | fwrite | 文件 |
fgetc
fgetc函数用于从文件中读取字符,其只有一个参数stream,为文件指针,指向字符来源的文件,函数成功执行返回值为获取字符的ASCII值,函数执行失败返回EOF。
fputc
fputc函数用于向文件输出字符,其有两个参数,第一个是要输出的字符,第二个是指针,指向字符输出到的文件,函数成功执行返回值是输出的字符,函数执行不成功返回EOF。
fgets
fgets函数用于从文件中读取一行文字,其有三个参数,第一个参数str为指针,指向存储文本行的起始地址,第二个参数是要读的文本行字节数(这个数量里包含了’\0’),第三个参数是文件指针,指向文本行源头的文件,当函数成功执行时,返回str所指向的地址,当执行失败是,返回NULL。
fputs
fputs函数用于输出文本行到文件,其有两个参数,第一个参数str指针,指向要输出的文本行,第二个参数是文件指针,指向文本行输出到的文件,函数成功执行返回一个非负值,函数执行失败返回EOF,值得注意的是文本行输出函数fputs输出文本行时,直到输出至’\0’才会停止,且’\0’不会是输出到文件。
fscanf
fscanf函数是用于从文件中读取格式化内容,其和scanf函数相比,只多出第一个参数stream,其为指针,指向读取的内容的源头,后面的参数和scanf一样的用法。
fprintf
fprintf函数是用于将格式化内容输出到文件,其和printf函数相比,同样只多出第一个参数stream,其为指针,指向输出内容所到的文件,其他参数和printf函数一样。
fread
fread函数是用于从二进制文件中读取内容到内存,其有四个参数,第一个参数指针ptr,指向用于存储读取内容的空间,第二个参数size是一次读取的内容占据多大空间,单位字节,第三个参数count是size大小的内容要读的次数,第四个参数stream是文件指针,指向读取内容的源头的文件,函数执行成功返回读取的次数count,当返回值和count不同时,函数执行失败。
fwrite
fwrite函数用于输出内容到二进制文件,其有四个参数,第一个参数指针ptr,指向要输出的内容,第二个参数size是输出的内容大小,第三个参数count是输出的内容次数,第四个参数文件stream,指向要输出的内容所到的文件,函数执行成功返回输出的次数count,如果返回值和count不相等,说明函数执行失败。
这里穿插两个函数进行对比
(1)sscanf函数和fscanf函数
先认识sscanf函数:
sscanf函数是用于从字符串中读取格式化内容,和fscanf函数相比,只有第一个参数不同,sscanf函数第一个参数为指针s,其指向读取内容的源头字符串,其他参数和scanf一致。
(2)sprintf函数和fprintf函数
先认识sprintf函数:
sprintf函数是将格式化内容输出到字符串,其和fprintf函数相比,只有第一个参数不同,sprintf第一个参数为指针str,其指向格式化内容所到的字符串,其他参数和printf一致。
feof和ferror
这两个函数是用于判断文件读取函数执行结束的原因,即:因为遇到文件末而结束或者读取错误而结束。
feof
feof函数是用于判断文件读取函数是否因为遇到文件末而结束,参数只有一个文件指针stream,指向导致文件读取函数结束的文件,如果是因为遇到文件末而结束返回非零值,否则返回0。
ferror
ferror函数用于判断文件读取函数是否因为读取错误而结束。参数只有一个文件指针stream,指向导致文件读取函数结束的文件,如果是因为遇到错误而结束返回非零值,否则返回0。
关于流
实际上文件指针就是流的一种,电脑还具备三个常开的流,即:标准输入流(stdin),标准输出流(stdout),标准错误输出流(stderr)。函数scanf,printf实际上就是默认使用标准输入流和标准输出流,在本质上scanf和fscanf,printf和fprintf是一致的。
定位文件指针实现随机读取
我们从文件中读取数据,不能总是按顺序读写,有时候我们只需要中间的某些内容,因此根据需要定位文件指针然后读取是很有必要的,
fseek
fseek函数是根据文件指针的位置和偏移量来定位文件指针。函数seek有三个参数,第一个参数是文件指针,指向读写的文件,第二个参数是当前文件指针的偏移量,第三个参数是文件指针偏移的参考位置,其共有三个取值,如下表所示:
origin | 参考位置 |
---|---|
SEEK_SET | 文件开头 |
SEEK_CUR | 文件指针的当前位置 |
SEEK_END | 文件结束 |
当函数成功执行时返回0,否则返回非零值。
举例说明:
int main()
{
FILE* pFile = fopen("example.txt", "wb");
if (pFile == NULL)
{
perror("fopen");
return 1;
}
fputs("This is an apple.", pFile);
fseek(pFile, 9, SEEK_SET);
fputs(" sam", pFile);
fclose(pFile);
return 0;
}
代码的意思为:打开一个文件,如果成功打开,输出文本行"This is an apple.”到文件,然后定位文件指针(将文件指针调整到相对文件开头9字节的地方),再输出文本行" sam"到文件,最后关闭文件。输出结果如下:
ftell
fseek函数是根据文件指针的位置和偏移量来定位指针,那么当前文件指针在什么位置我们如何知晓呢,ftell函数就是告诉我们当前文件指针相对文件开头的偏移量的,先认识函数: