目录
1. 为什么使用文件
当我们写程序并运行时,会把一些信息存到内存中,可一旦退出程序,信息就被销毁,下次运行程序时还要重新输入,这就很难受了!
我们想要只有我们自己选择删除数据的时候,数据才不复存在。
这就涉及到了数据持久化的问题,我们一般数据持久化的方法有,把数据存放在磁盘文件、存放到数据库等方式。
使用文件我们可以将数据直接存放在电脑的硬盘上,做到了数据的持久化。
2. 什么是文件
磁盘上的文件是文件。
3. 文件的打开和关闭
3.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;
不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异。
每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者不必关心细节。
一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便。
下面我们可以创建一个FILE*的指针变量:
FILE* pf;//文件指针变量
定义pf是一个指向FILE类型数据的指针变量。可以使pf指向某个文件的文件信息区(是一个结构体变量)。通过该文件信息区中的信息就能够访问该文件。也就是说,通过文件指针变量能够找到与它关联的文件。如图:
3.2 文件的打开和关闭
文件在读写之前应该先打开文件,在使用结束之后应该关闭文件。
ANSIC 规定使用fopen函数来打开文件,fclose来关闭文件。
打开方式如下:
文件使用方式 | 含义 | 如果指定文件不存在 |
“r”(只读) | 为了输入数据,打开一个已经存在的文本文件 | 出错 |
“w”(只写) | 为了输出数据,打开一个文本文件 | 建立一个新的文件 |
“a”(追加) | 向文本文件尾添加数据 | 建立一个新的文件 |
“rb”(只读) | 为了输入数据,打开一个二进制文件 | 出错 |
“wb”(只写) | 为了输出数据,打开一个二进制文件 | 建立一个新的文件 |
“ab”(追加) | 向一个二进制文件尾添加数据 | 出错 |
“r+”(读写) | 为了读和写,打开一个文本文件 | 出错 |
“w+”(读写) | 为了读和写,建议一个新的文件 | 建立一个新的文件 |
“a+”(读写) | 打开一个文件,在文件尾进行读写 | 建立一个新的文件 |
“rb+”(读写) | 为了读和写打开一个二进制文件 | 出错 |
“wb+”(读写) | 为了读和写,新建一个新的二进制文件 | 建立一个新的文件 |
“ab+”(读写) | 打开一个二进制文件,在文件尾进行读和写 | 建立一个新的文件 |
举例:
#include <stdio.h>
int main ()
{
FILE * pFile;
//打开文件
pFile = fopen ("myfile.txt","w");
if (pFile == NULL)//检查文件打开是否有问题
{
perror("fopen");
return 1;
}
//文件操作
//关闭文件
fclose(pFile);
pFlie = NULL;
return 0;
}
4. 文件的顺序读写
功能 | 函数名 | 适用于 |
字符输入函数 | fgetc | 所有输入流 |
字符输出函数 | fputc | 所有输出流 |
文本行输入函数 | fgets | 所有输入流 |
文本行输出函数 | fputs | 所有输出流 |
格式化输入函数 | fscanf | 所有输入流 |
格式化输出函数 | fprintf | 所有输出流 |
二进制输入 | fread | 文件 |
二进制输出 | fwrite | 文件 |
挨个举例:
4.1 fgetc && fputc
4.2 fgets && fputs
4.3 fscanf && fread
4.4 fread && fwrite
4.5 对比一组函数:
sscnaf && sprintf 举例:
int main()
{
char buf[256] = { 0 };
struct S s = { "zhangsan", 20, 95.5 };
struct S tmp = { 0 };
sprintf(buf, "%s %d %lf", s.name, s.age, s.d);
printf("%s\n", buf);//字符串
//从buf字符串中提取结构体数据
sscanf(buf, "%s %d %lf", tmp.name, &(tmp.age), &(tmp.d));
printf("%s %d %lf", tmp.name, tmp.age, tmp.d);//格式化的形式
return 0;
}
5. 文件的随机读写
前面讲到fgetc函数,如果多次从文件得到字符,字符会按照顺序被取出,但是实际中我们也许不想按顺序得到字符,而是按照需求去取得,这样,就不得不介绍文件随机读写涉及的函数。
5.1 fseek
举个例子:
#include <stdio.h>
int main()//实现创建一个test.txt并在里面放入abcdef(不创建"r"读取出出错)
{
//打开文件
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//随机读
int ch = fgetc(pf);//a
printf("%c\n", ch);
ch = fgetc(pf);//b 因为每读取一个字符后自动跳下一个,所以读完'a' 是 'b'
printf("%c\n", ch);
fseek(pf, 4, SEEK_SET);//起始位置偏移4个字符,指向e
ch = fgetc(pf);//e
printf("%c\n", ch);
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
5.2 ftell && rewind
ftell :返回文件指针相对于起始位置的偏移量
rewind: 让文件指针的位置回到文件的起始位置
long int ftell ( FILE * stream );
void rewind ( FILE * stream );
举个例子:
int main()
{
//打开文件
FILE* pf = fopen("test.txt", "w");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//随机读
fputc('a', pf);
fputc('b', pf);
fputc('c', pf);
fputc('d', pf);
fseek(pf, -3, SEEK_CUR);
fputc('w', pf);//此时文件内容为awcd
long pos = ftell(pf);//此时相对于起始位置偏移量为2
printf("%ld\n", pos);
rewind(pf);//回到起始位置
pos = ftell(pf);//此时偏移量为0
printf("%ld\n", pos);//0
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}