文章目录
前言
提示:以下是本篇文章正文内容,下面案例可供参考
🌟一、为什么使用文件
我们前面学习结构体时,写通讯录的程序,当通讯录运行起来的时候,可以给通讯录中增加、删除数据,此时数据是存放在内存中,当程序退出的时候,通讯录中的数据自然就不存在了,等下次运行通讯录程序的时候,数据又得重新录入,如果使用这样的通讯录就很难受。
我们在想既然是通讯录就应该把信息记录下来,只有我们自己选择删除数据的时候,数据才不复存在。这就涉及到了数据持久化的问题,我们一般股教据持久化的方法有,把数据存放在磁盘文件、存放到数据库等方式。
使用文件我们可以将数据直接存放在电脑的硬盘上,做到了数据的持久化。
🌟二、什么是文件
🌏2.1程序文件:
🌏2.2数据文件:
🌏2.3文件名:
🌟三、文件的打开和关闭
🌏3.1文件指针:
缓冲文件系统中,关键的概念是"文件类型指针",简称“文件指针"。
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。声明的,取名FILE.
不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异。
每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者不必关心细节。
一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便。下面我们可以创建一个FILE的指针变量:*
FILE* pf;//文件指针变量
定义pf是一个指向FILE类型数据的指针变量。可以使pf指向某个文件的文件信息区(是一个结构体变量)。通过该文件信息区中的信息就能够访问该文件。也就是说,通过文件指针变量能够找到与它关联的文件。
🌏3.2文件的打开和关闭:
//打开文件
FILE * fopen ( const char * filename, const char * mode );//关闭文件
int fclose ( FILE * stream );
文件使用方式 | 含义 | 如果指定文件不存在 |
---|---|---|
“r”(只读) | 为了输入数据,打开一个已经存在的文本文件 | 出错 |
"w”(只写) | 为了输出数据,打开一个文本文件 | 建立一个新的文件 |
“a”(追加) | 向文本文件尾添加数据 | 建立一个新的文件 |
“rb”(只读) | 为了输入数据,打开一个二进制文件 | 出错 |
“wb”(只写) | 为了输出数据,打开一个二进制文件 | 建立一个新文件 |
“ab”(追加) | 向一个二进制文件尾添加数据 | 建立一个新的文件 |
"r+”(读写) | 为了读和写,打开一个文本文件 | 出错 |
“w+”(读写) | 为了读和写,建议一个新的文件 | 建立一个新的文件 |
"a+”(读写) | 打开一个文件,在文件尾进行读写 | 建立一个新的文件 |
“rb+”(读写) | 为了读和写打开一个二进制文件 | 出错 |
"wb+”(读写) | 为了读和写,新建一个新的二进制文件 | 建立一个新的文件 |
"ab+”(读写) | 打开一个二进制文件,在文件尾进行读和写 | 建立一个新的文件 |
#include<stdio.h>
int main()
{
FILE* pf = fopen("test.txt", "w");
if (pf == NULL)
{
perror("fopen");
return 1;
}
fclose(pf);
pf = NULL;
return 0;
}
🌟四、文件的顺序读写
🌏3.1顺序读写函数介绍
功能 | 函数名 | 适用于 |
---|---|---|
字符输入函数 | fgetc | 所有输入流 |
字符输出函数 | fputc | 所有输出流 |
文本行输入函数 | fgets | 所有输入流 |
文本行输出函数 | fputs | 所有输出流 |
格式化输入函数 | fscanf | 所有输入流 |
格式化输出函数 | fprintf | 所有输出流 |
二进制输入 | fread | 文件 |
二进制输出 | fwrite | 文件 |
🌏3.2fputc函数 — 写文件
假设我们在这个test.txt中以写的方式写入26个英文字母
#include<stdio.h>
int main()
{
//打开文件
FILE* pf = fopen("test.txt", "w");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//写文件
int i = 0;
for (i = 0; i < 26; i++)
{
fputc('a'+i, pf);
}
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
🌏3.3fgetc函数 ---- 读文件
//fgetc---读文件
#include<stdio.h>
int main()
{
//打开文件
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//读文件
int i = 0;
for (i = 0; i < 26; i++)
{
int ch = fgetc(pf);//返回的是ASCII码值读错返回EOF(-1)
printf("%c ", ch);
}
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
🌏3.4fputs函数 ---- 写文件
#include<stdio.h>
int main()
{
//打开文件
FILE* pf = fopen("test.txt", "w");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//写文件
fputs("hello world", pf);
fputs("abcdef", pf);
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
🌏3.5fgets函数 ---- 读文件
#include<stdio.h>
int main()
{
//打开文件
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//读文件
char arr[20];
fgets(arr,5,pf);
printf("%s\n", arr);
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
🌏3.6fprintf函数 ---- 写文件
#include<stdio.h>
struct S
{
int n;
float f;
char arr[20];
};
int main()
{
struct S s = { 100,3.14,"lisi" };
//打开文件
FILE* pf = fopen("test.txt", "w");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//写文件
fprintf(pf,"%d %f %s",s.n ,s.f ,s.arr );
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
🌏3.7fscanf函数 ---- 读文件
#include<stdio.h>
struct S
{
int n;
float f;
char arr[20];
};
int main()
{
struct S s = { 0 };
//打开文件
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
perror("fopen");
return 1;
}
//写文件
fscanf(pf, "%d %f %s", &(s.n), &(s.f), s.arr);//arr数组名不用取地址
printf("%d %f %s\n", s.n, s.f, s.arr);
//关闭文件
fclose(pf);
pf = NULL;
return 0;
}
🌏3.8关于输入流输出流
🌏3.9对比一组函数
#include<stdio.h>
struct S
{
int n;
float f;
char arr[20];
};
int main()
{
//序列化和反序列化的时候可能用到
struct S s = { 20,3.5f,"wangwu" };
//把一个结构体转换成字符串
char arr[200] = { 0 };
sprintf(arr,"%d %f %s\n", s.n, s.f, s.arr);
printf("字符串的数据:%s\n",arr);
//把一个字符串转换成对应格式化数据
struct S tmp = { 0 };
sscanf(arr, "%d %f %s\n", &(tmp.n), &(tmp.f), tmp.arr);
printf("格式化的数据:%d %f %s\n", tmp.n, tmp.f, tmp.arr);
return 0;
}
🌏3.10二进制写文件—fwrite
//二进制写文件 --- fwrite
#include<stdio.h>
struct S
{
char name[20];
int age;
float score;
};
int main()
{
struct S s = { "zhangsan",19,99.5 };
FILE* pf = fopen("text.dat", "wb");
if (pf == NULL)
{
perror("fopen");
return 1;
}
fwrite(&s, sizeof(struct S), 1, pf);
fclose(pf);
pf = NULL;
return 0;
}
🌏3.11二进制读文件—fread
//二进制的读文件---fread
#include<stdio.h>
struct S
{
char name[20];
int age;
float score;
};
int main()
{
struct S s = { "zhangsan",19,99.5 };
FILE* pf = fopen("text.dat", "rb");
if (pf == NULL)
{
perror("fopen");
return 1;
}
fread(&s, sizeof(struct S), 1, pf);
printf("%s %d %f\n", s.name, s.age, s.score);
fclose(pf);
pf = NULL;
return 0;
}
🌏3.12根据上面所学改造一下通讯录
🌟五、文件的随机读写
🌏4.1fseek函数
根据文件指针的位置和偏移量来定位文件指针。
举个例子:我们在文件中放入abcdef,我们先读取4次读到d的位置,此时指针指向了e这时,我们希望读到b的位置显然就需要用到我们的fseek函数我们可以有三种方式从文件的开始位置,文件指针的当然位置和文件的末尾。
#include<stdio.h>
int main()
{
FILE* pf = fopen("text.dat", "r");
if (pf == NULL)
{
perror("fopen");
return 1;
}
int ch=fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);//当时读到的位置是d的位置指针指向e
//如果我们想读到b我们需要让指针偏移3个位置
//fseek(pf, -3,SEEK_CUR);
fseek(pf, 1, SEEK_SET);
ch = fgetc(pf);
printf("%c\n", ch);
fclose(pf);
pf = NULL;
return 0;
}
🌏4.2ftell函数
返回文件指针相对于起始位置的偏移量
#include<stdio.h>
int main()
{
FILE* pf = fopen("text.dat", "r");
if (pf == NULL)
{
perror("fopen");
return 1;
}
int ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
printf("%d\n", ftell(pf));
fclose(pf);
pf = NULL;
return 0;
}
🌏4.2rewind函数
让文件指针的位置回到文件的起始位置
#include<stdio.h>
int main()
{
FILE* pf = fopen("text.dat", "r");
if (pf == NULL)
{
perror("fopen");
return 1;
}
int ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
rewind(pf);//我们这里使用rewind函数使文件指针回到文件起始位置
printf("%d\n", ftell(pf));//所以这里的偏移量为0
ch = fgetc(pf);//我们再次开始读,读到的就是a
printf("%c\n", ch);
fclose(pf);
pf = NULL;
return 0;
}
🌟六、文本文件和二进制文件
🌟七、文本读取结束的判定
o fgetc判断是否为EOF.
o fgets判断返回值是否为NULL.
o fread判断返回值是否小于实际要读的个数。
读取失败返回EOF
1.遇到文件末尾,返回EOF,同时设置一个状态,遇到文件末尾了,使用feof来检测这个状念
⒉.遇到错误,返回EOF,同时设置一个状态,遇到了错误,使用ferror来检测这个状态