为什么使用文件
写通讯录程序时,当通讯录运行起来的时候,可以给通讯录中增加、删除数据,此时数据存放在内存中,当程序退出的时候,通讯录中的数据自然就不存在了,等下次运行通讯录程序的时候,数据又得重新录入,如果使用这样的通讯录,那么每次运行通讯录程序时都需要重新录入,会很麻烦。因此通讯录就应该把信息记录下来,只有主动删除数据时,数据才会不复存在,这就涉及数据持久化的问题。
一般数据持久化的方式有:把数据存放在磁盘文件、存放到数据库等方式。
什么是文件
磁盘上的文件是文件 :当把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中,这时候处理的就是磁盘上文件。
不过程序设计中,从功能角度来看,一般文件分为两种:程序文件、数据文件
(1)程序文件:windows环境中的源程序文件(.c)、目标文件(.obj)、可执行程序(.exe)。
(2)数据文件:文件内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件。
本博客主要讲述数据文件。
文件的打开和关闭
文件名作为文件的唯一标识,包含3部分:文件路径+文件名主干+文件后缀
如:C:\code\test.txt
另外,程序员应有的安全感就是把文件扩展名显示出来
文件的名称是以显示文件扩展名为准的,如不勾选文件扩展名时,文件名称为file.txt
如果勾选文件扩展名,那么文件名称为file.txt.txt,这时候如果想打开file.txt文件,那么无论如何都会失败。
(1)文件类型指针
简称“文件指针”,每个被使用的文件,都在内存中开辟了相应的文件信息区,用来存放文件的相关信息(包括文件名、文件状态、文件位置等)。这些信息保存在FILE结构体变量中,FILE结构体类型是系统声明的。
1. struct _iobuf 2. { 3. char* _ptr; 4. int _cnt; 5. char* _base; 6. int _flag; 7. int _file; 8. int charbuf; 9. int _bufsiz; 10. char* _tmpfname; 11. }; 12. typedef struct _iobuf FILE;
不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异。
每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充相应信息,使用时不必关心细节。
一般都是通过一个FILE的的指针来维护FILE结构变量,使用起来更加方便。
创建FILE*的指针变量:
FILE* pf;//文件指针变量
这就定义了pf是一个指向FILE类型数据的指针变量,可以使pf指向某个文件的文件信息区(是一个结构体变量)。通过文件信息区中的信息就能够访问该文件,即通过文件指针变量能够找到与它关联的文件。
(2)文件的打开和关闭
文件在读写之前,应该先打开文件,在使用结束之后应该关闭文件。
在写程序时,打开文件的同时,都会返回一个FILE指针变量,指向该文件,这就机那里了指针和文件的关系。
ANSIC规定使用fopen函数来打开文件,fclose来关闭文件。
1. //打开文件 2. FILE* open(const char* filename, const char* mode); 3. //关闭文件 4. int close(FILE * stream);
程序和文件读取、写入示意图:
打开方式有多种:
文件使用方式 |
含义 |
如果指定文件不存在 |
“r”(只读) |
为了使用数据,打开一个已经存在的文本文件 |
出错 |
“w”(只写) |
为了输出数据,打开一个文本文件 |
建立一个新的文件 |
“a”(追加) |
向文本文件末尾添加数据 |
出错 |
“rb”(只读) |
为了输入数据,打开一个二进制文件 |
出错 |
“wb”(只写) |
为了输出数据,打开一个二进制文件 |
建立一个新的文件 |
“ab”(追加) |
像一个二进制文件末尾添加数据 |
出错 |
“r+”(读写) |
为了读和写,打开一个文本文件 |
出错 |
“w+”(读写) |
为了读和写,建立一个新的文件 |
建立一个新的文件 |
“a+”(读写) |
打开一个文件,在文件末尾进行读写 |
建立一个新的文件 |
“rb+”(读写) |
为了读和写打开一个二进制文件 |
出错 |
“wb+”(读写) |
为了读和写,新建一个新的二进制文件 |
建立一个新的文件 |
“ab+”(读写) |
打开一个二进制文件,在文件末尾进行读和写 |
建立一个新的文件 |
在VS2019 解决方案下手动创建文件test.data
1. #define _CRT_SECURE_NO_WARNINGS 1 2. #include<stdio.h> 3. 4. int main() 5. { 6. //打开文件 7. FILE* pf = fopen("test.data", "r"); 8. 9. //文件操作 10. if (pf == NULL) 11. { 12. perror("fopen"); 13. return 1; 14. } 15. fputs("fopen test", pf); 16. //关闭文件 17. fclose(pf); 18. pf = NULL; 19. return 0; 20. }
如果使用绝对路径打开文件,向test.data文件里面写东西
1. #define _CRT_SECURE_NO_WARNINGS 1 2. #include<stdio.h> 3. 4. int main() 5. { 6. //打开文件 7. FILE* pf = fopen("C:\Users\Delia\source\repos\bit4\test.data", "w");//使用绝对路径打开文件 8. 9. //文件操作 10. if (pf == NULL) 11. { 12. perror("fopen"); 13. return 1; 14. } 15. fputs("fopen test", pf); 16. //关闭文件 17. fclose(pf); 18. pf = NULL; 19. return 0; 20. } 21.
路径会报错:
这是因为路径没有使用转义字符,会把路径中的\r 、\b、\t当做转义字符来处理,路径就找不到了,路径需要加上转义字符
1. #define _CRT_SECURE_NO_WARNINGS 1 2. #include<stdio.h> 3. 4. int main() 5. { 6. //打开文件 7. FILE* pf = fopen("C:\\Users\\Delia\\source\\repos\\bit4\\test.data", "w");//路径加上转义字符 8. 9. //文件操作 10. if (pf == NULL) 11. { 12. perror("fopen"); 13. return 1; 14. } 15. fputs("fopen test", pf); 16. //关闭文件 17. fclose(pf); 18. pf = NULL; 19. return 0; 20. }
ctrl+F5运行之后, 打开test.data文件,写入成功: