C语言-文件系统详解(一)

简介: C语言-文件系统详解

为什么使用文件

写通讯录程序时,当通讯录运行起来的时候,可以给通讯录中增加、删除数据,此时数据存放在内存中,当程序退出的时候,通讯录中的数据自然就不存在了,等下次运行通讯录程序的时候,数据又得重新录入,如果使用这样的通讯录,那么每次运行通讯录程序时都需要重新录入,会很麻烦。因此通讯录就应该把信息记录下来,只有主动删除数据时,数据才会不复存在,这就涉及数据持久化的问题。

一般数据持久化的方式有:把数据存放在磁盘文件、存放到数据库等方式。


什么是文件

磁盘上的文件是文件 :当把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中,这时候处理的就是磁盘上文件。

不过程序设计中,从功能角度来看,一般文件分为两种:程序文件、数据文件

(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文件,写入成功:


相关文章
|
存储 C语言
|
存储 C语言
|
存储 Linux C语言
【C语言有什么用?①】从零开始撸一个用户态模拟文件系统
【C语言有什么用?①】从零开始撸一个用户态模拟文件系统
【C语言有什么用?①】从零开始撸一个用户态模拟文件系统
|
15天前
|
程序员 C语言
C语言库函数 — 内存函数(含模拟实现内存函数)
C语言库函数 — 内存函数(含模拟实现内存函数)
24 0
|
26天前
|
编译器 C语言 C++
【C语言】memset()函数(内存块初始化函数)
【C语言】memset()函数(内存块初始化函数)
26 0
|
26天前
|
编译器 C语言 C++
【C语言】memcpy()函数(内存块拷贝函数)
【C语言】memcpy()函数(内存块拷贝函数)
42 0
|
27天前
|
C语言 C++
【C语言】rand()函数(如何生成指定范围随机数)
【C语言】rand()函数(如何生成指定范围随机数)
16 0
|
2天前
|
存储 C语言
C语言函数的返回值
C语言函数的返回值
7 0
|
2天前
|
C语言 Windows
C语言中的fopen与fclose函数详解
C语言中的fopen与fclose函数详解
10 1
|
2天前
|
C语言
深入理解C语言中的printf函数及数据输出
深入理解C语言中的printf函数及数据输出
11 0