持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第11天 @TOC
一、为什么要使用文件?
在我之前写的 : 动态通讯录中未引入文件管理,我们在使用通讯录中,可以进行数据的修改,但数据只是放在内存中的,只能在本次运行中保存数据,下次使用时,数据就消失不见了。 只有使用文件,我们就可以将数据放在电脑硬件中,做到了数据的持久化。
二、文件是什么?
文件一般从功能分为两种:程序文件,数据文件
1.程序文件
源文件:(后缀为.c) 目标文件:(windows环境后缀为.obj) 可执行程序(windows环境后缀为.exe)
2.数据文件
文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件
3.文件名
一个文件要有一个唯一的文件标识,以便用户识别和引用。 文件名包含3部分:文件路径+文件名主干+文件后缀 例如: D:\code\java.txt 为了方便起见,文件标识常被称为文件名。
三、文件的打开和关闭
1.文件指针
每一个被使用的文件都在内存中开辟一个相应的文件信息区,用来存放文件的相关信息(文件名,文件状态及文件当前位置等)。
一般在打开一个文件时,系统会根据文件自动创建一个FILE的变量。 一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样比较方便。
代码如下(示例):
FILE *pf;//文件指针变量
我们创建的pf是指向FILE类型数据的指针变量,可以使PF指向文件信息区,以达到访问该文件的目的。
2.文件如何打开及关闭?
文件在每次读写之前,应首先打开文件,使用完之后及时关闭文件。
ANSIC规定使用fopen函数打开文件
//打开文件
FILE* fopen(const char* filename,const char * mode);
ANSIC规定使用fclose函数关闭文件
//关闭文件
int fclose(FILE* stream);
| 文件使用方式 | 含义 | 如果指定文件不存在 |
|---|---|---|
| "r"(只读) | 为了输入数据,打开一个已存在的文本文件 | 出错 |
| "w"(只写) | 为了输出数据,打开一个文本文件 | 建立一个新的文件 |
| "a"(追加) | 向文本尾添加数据 | 建立一个新的文件 |
| "rb"(只读) | 为了输入数据,打开一个二进制文件 | 出错 |
| "wb"(只写) | 为了输出数据,打开一个二进制文件 | 建立一个新的文件 |
| "ab"(追加) | 向一个二进制文件尾添加数据 | 出错 |
| "r+"(读写) | 为了读和写,打开一个文本文件 | 出错 |
| "w+"(读写) | 为了读和写,建立一个新的文件 | 建立一个新的文件 |
| "a+"(读写) | 打开一个文件,在文件尾进行读写 | 建立一个新的文件 |
| "rb+"(读写) | 为了读和写打开一个二进制文件 | 出错 |
| "wb+"(读写) | 为了读和写,建立一个新的二进制文件 | 建立一个新的文件 |
| "ab+"(读写) | 打开一个二进制文件,在文件尾进行读写 | 建立一个新的文件 |
代码如下(示例):
int main()
{
FILE* pf;
//打开文件
pf = fopen("dachang.txt", "w");
//文件操作
if (pf != NULL)
{
fputs("woyaojindachang", pf);
//关闭文件
fclose(pf);
}
return 0;
}
四、文件的顺序读写
| 功能 | 函数名 | 适用于 |
|---|---|---|
| 字符输入函数 | fgetc | 所有输入流 |
| 字符输出函数 | fputc | 所有输出流 |
| 文本行输入函数 | fgets | 所有输入流 |
| 文本输出函数 | fputs | 所有输出流 |
| 格式化输入函数 | fscanf | 所有输出流 |
| 格式化输出函数 | fprintf | 所有输出流 |
| 二进制输入 | fread | 文件 |
| 二进制的输出 | fwrite | 文件 |
1.fgetc
//写一个字符
int main()
{
FILE* pf = fopen("test.txt", "w");
if (pf == NULL)
{
perror("fopen\n");
return 1;
}
//写文件
char i = 0;
for (i = 'a'; i <= 'z'; i++)
{
fputc(i, pf);
}
fclose(pf);
pf = NULL;
return 0;
}
2.fputc
//读一个字符
int main()
{
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
perror("fopen\n");
return 1;
}
//读文件
char ch = 0;
while ((ch = fgetc(pf)) != EOF)
{
printf("%c ", ch);
}
fclose(pf);
pf = NULL;
return 0;
}
3.fgets
//写一行数据
int main()
{
FILE* pf = fopen("test.txt", "w");
if (pf == NULL)
{
perror("fopen\n");
return 1;
}
//写一行数据
fputs("woyaojindachang\n", pf);
fclose(pf);
pf = NULL;
return 0;
}
4.fputs
//读一行数据
int main()
{
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
perror("fopen\n");
return 1;
}
//读一行数据
char ch[20];
fgets(ch, 20, pf);
printf("%s", ch);
fclose(pf);
pf = NULL;
return 0;
}
5.fprintf
struct S
{
char arr[10];
int age;
float score;
};
int main()
{
struct S s = { "lisi",18,100.0f};
FILE* pf = fopen("text.txt", "w");
if (pf == NULL)
{
perror("fopen");
return 1;
}
fprintf(pf, "%s %d %f", s.arr, s.age, s.score);
return 0;
}
6.fscanf
struct S
{
char arr[10];
int age;
float score;
};
int main()
{
struct S s = { 0 };
FILE* pf = fopen("text.txt", "r");
if (pf == NULL)
{
perror("fopen");
return 1;
}
fscanf(pf, "%s %d %f", s.arr, &(s.age), &(s.score));
printf("%s %d %f\n", s.arr, s.age, s.score);
return 0;
}
7.fwrite
struct S
{
char arr[10];
int age;
float score;
};
int main()
{
struct S s = { "zhangsan", 25, 50.5f };
//以二进制的形式写到文件中
FILE* pf = fopen("test.txt", "wb");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//二进制的方式写
fwrite(&s, sizeof(struct S), 1, pf);
fclose(pf);
pf = NULL;
return 0;
}
这里打开文件乱码,正常,这里使二进制文件
8.fread
struct S
{
char arr[10];
int age;
float score;
};
int main()
{
struct S s = { 0 };
//以二进制的形式写到文件中
FILE* pf = fopen("test.txt", "rb");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//二进制的方式读
fread(&s, sizeof(struct S), 1, pf);
printf("%s %d %f\n", s.arr, s.age, s.score);
fclose(pf);
pf = NULL;
return 0;
}