1. fgetc和fputc
上文图示用到这两个函数
代码如下:
//这是fgetc函数,可以理解为得到文件中的一个字符 int main() { FILE* pf = fopen("test.txt", "r"); for (int i = 0; i < 10; i++) { char src = fgetc(pf); printf("%c", src); } fclose(pf); pf = NULL; return 0; } //这是fputc函数,可以理解为放置文件中一个字符 int main() { FILE* pf = fopen("test.txt", "w"); for (int i = 0; i < 26; i++) { fputc('a' + i, pf); } fclose(pf); pf = NULL; return 0; }
实际上,fgetc和fputc函数就是得到或放置一个字符在文件中,返回值为int
2. fgets和fputs
fgets和fputs,可以理解为得到或者放置一个字符串在文件中。
还有一个特点,如图所示:
fgets如果没有读取到内容,返回NULL
代码如下:
//fgets函数 int main() { FILE* pf = fopen("test.txt", "r"); char arr[20]; while (fgets(arr, 20, pf) != NULL) { printf("%s", arr); } fclose(pf); pf = NULL; return 0; } //fputs函数 int main() { FILE* pf = fopen("test.txt", "w"); char arr[20]; fputs("hello world!!\n", pf); fputs("hello why\n", pf); fputs("hello fzx\n", pf); fclose(pf); pf = NULL; return 0; }
3. fscanf和fprintf
实际上和scanf、printf差不多的形式,只是多一个参数 FILE*stream
如图:
如果fscanf读取的时候没有数据,那么也不会报错,就认为是没有读取吧
代码如图所示:
struct S { char name[20]; int age; double score; }; //fscanf格式化读取pf流中的数据,赋值给后面的参数,和scanf用法差不多, //只是多一个FILE*类型,后面于scanf用法一致,改用&就用 int main() { FILE* pf = fopen("test.txt", "r+"); //读写都行 //格式化输出 //struct S s = { "why",19,100 }; struct S s = {0}; struct S s1 = { 0 }; fscanf(pf, "%s %d %lf\n", s.name, &(s.age), &(s.score)); printf("%s %d %lf\n", s.name, (s.age), (s.score)); //fscanf(pf, "%s %d %lf\n", s1.name, &(s1.age), &(s1.score)); //printf("%s %d %lf\n", s1.name, (s1.age), (s1.score)); int a = 10; /*fscanf(pf, "%d", a); printf("%d\n", a);*/ //这是进行测试如果没有相应的数据读取的时候,不会报错,只是相当于没有这一行代码,不会发生任何改变(对相应a) fclose(pf); pf = NULL; return 0; } //这是fprintf函数的使用 int main() { FILE* pf = fopen("test.txt", "r+"); //读写都行 //格式化输出 struct S s = { "why",19,100 }; fprintf(pf, "%s %d %lf\n", s.name, (s.age), (s.score)); //printf("%s %d %lf\n", s.name, (s.age), (s.score)); fclose(pf); pf = NULL; return 0; }
1.fscanf 格式化读取pf流中的数据,赋值给后面的参数,和 scanf 用法差不多,只是多一个FILE*类型,后面于 scanf 用法一致,该用&就用
2.fprintf 格式化写入文件中数据,上面代码中,将结构体变量s,数据,写入pf管理的文件中
3.fscanf 如果没有相应的数据读取的时候,不会报错,只是相当于没有这一行代码,不会发生任何改变(对相应a)
4. fread和fwrite
只能接收文件流,二进制文件,可能会有乱码产生
如图分析:
代码演示:
struct S { char name[20]; int age; double score; }; //先写入二进制文件 fwrite int main() { struct S s = { "fzx",18,100 }; FILE* pf = fopen("test.txt", "wb"); //写一个二进制文件 fwrite(&s, sizeof(struct S), 1, pf); fclose(pf); pf = NULL; return 0; } //再读取二进制文件 fread int main() { struct S s = { 0 }; FILE* pf = fopen("test.txt", "rb"); //读取文件 fread(&s, sizeof(struct S), 1, pf); printf("%s %d %lf\n", s.name, s.age, s.score); return 0; }
5. 对比一组函数
scanf / fscanf / sscanf
printf / fprintf /sprintf
前面四种大家都认识了,那么sscanf和sprintf是什么呢?
如图所示:
3.4 文件的随机读写
1. fseek
根据文件指针的位置和偏移量来定位文件指针。(改变光标位置)
代码演示:
int main() { FILE* pFile; pFile = fopen("example.txt", "wb"); fputs("This is an apple.", pFile); fseek(pFile, 9, SEEK_SET); fputs(" sam", pFile); fclose(pFile); return 0; }
2.ftell
返回文件指针相对于起始位置的偏移量
代码演示:
int main() { FILE* pFile; pFile = fopen("example.txt", "wb"); int num = ftell(pFile); printf("%d ", num); return 0; }
3.rewind
让文件指针的位置回到文件的起始位置
图文演示:
代码演示:
int main() { FILE* pf = fopen("test.txt", "w"); for (int i = 0; i < 10; i++) { fputc('a' + i, pf); } int num = ftell(pf); printf("%d \n", num); rewind(pf); printf("%d \n", ftell(pf)); fclose(pf); pf = NULL; return 0; }
3.5 文本文件和二进制文件
1.根据数据的组织形式,数据文件被称为文本文件或者二进制文件。
2.数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。
3.如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储 的文件就是文本文件。
字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。
测试文本文件和二进制文件代码:
int main() { int a = 10000; FILE* pf = fopen("test.txt", "wb"); fwrite(&a, 4, 1, pf);//二进制的形式写到文件中 fclose(pf); pf = NULL; return 0; }
3.6文件读取结束的判定
在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束,而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。
1.文本文件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets )
fgetc 判断是否为 EOF .
fgets 判断返回值是否为 NULL
2. 二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。
fread判断返回值是否小于实际要读的个数
图示:
代码演示:
//文本文件 int main(void) { int c; // 注意:int,非char,要求处理EOF FILE* fp = fopen("test.txt", "r"); if(!fp) { perror("File opening failed"); return EXIT_FAILURE; } //fgetc 当读取失败的时候或者遇到文件结束的时候,都会返回EOF while ((c = fgetc(fp)) != EOF) // 标准C I/O读取文件循环 { putchar(c); }//判断是什么原因结束的 if (ferror(fp)) puts("I/O error when reading"); else if (feof(fp)) puts("End of file reached successfully"); fclose(fp); }
总结
主要讲解文件操作的的一些函数,fopen、fclose、fgets、fwrite等等函数的介绍,以及对于文件的分类、输入和输出流的分析,以及更多文件操作的函数图示分解。
下文我们将使用文件操作,升级通讯录。