1.读写文件的其他操作
对于文件操作还有以下的几个函数:
函数名 | 功能 | 适用性 |
fscanf() | 格式化输入函数 | 所有输入流 |
fprintf() | 格式化输出函数 | 所有输出流 |
fread() | 二进制输入 | 文件 |
fwrite() | 二进制输出 | 文件 |
让我们一个一个来分析:
(1)fscanf函数
先看一下官网对其的解释:
该函数的第一个参数是一个文件指针,第二个参数format是格式字符串,用于指定要读取的数据的格式;第三个参数...是要读取的数据的地址,它的返回值为成功匹配和读取的数据项数量。
我们直接使用实例来加深对其的理解:
#include<stdio.h> struct Student { char name[20]; int age; }; int main() { struct Student stu = { 0 }; FILE* pf = fopen("test.txt", "r"); if (pf == NULL) { perror("fopen"); return 1; } //使用fscanf函数格式化将文件中内容输出 fscanf(pf, "%s %d", stu.name, &(stu.age)); printf("%s %d", stu.name, stu.age); fclose(pf); pf = NULL; return 1; }
这样我们就完成了数据的格式化输出。
(2)fprintf函数
先看一下官网对其的解释:
该函数的第一个参数是文件指针,第二个参数format是格式字符串,用于指定要写入的数据的格式,第三个参数...是要写入的数据,fprintf函数会根据格式字符串的指定,将数据按指定格式写入到文件中,它的返回值为成功写入的字符数量。
我们直接使用实例来加深对其的理解:
#include<stdio.h> struct Student { char name[20]; int age; }; int main() { struct Student stu = { "zhangsan",20}; FILE* pf = fopen("test.txt", "w"); if (pf == NULL) { perror("fopen"); return 1; } //使用fprintf函数格式化将内容写入文件 fprintf(pf, "%s %d", stu.name, stu.age); fclose(pf); pf = NULL; return 1; }
这样我们就完成了数据的格式化输入。
补充:
区分printf 、fprintf 、 sprintf函数 + 区分scanf 、fscanf 、sscanf函数
让我们先来看printf 、fprintf 、 sprintf这一组,先看一下这几个函数的定义:
printf函数:
fprintf函数:
sprintf函数:
根据之前所学习的fprintf函数的几个参数,我们可以知道:
printf:将数据格式化的输出在标准输出流;
fprintf:将数据格式化的输出在任意指定的输出流中;
sprintf:将数据格式化转换为字符串
这里就不在对printf和fprintf函数进行解释了(上文中已解释),只对sprintf函数进行解释:
我们直接使用案例讲解:
#include<stdio.h> struct Student { char name[10]; int age; }; int main() { struct Student stu = { "zhangsan",20 }; char str[20] = { 0 }; //将stu结构体中的数据格式化转换为字符串后储存到str数组中 sprintf(str, "%s %d", stu.name, stu.age); printf("%s", str); return 0; }
从结果我们可以看出,sprintf函数将结构体中的数据格式化的转换成了字符串并储存到了str中。
区分完了printf 、fprintf 、 sprintf函数的区别,下面在让我们区分一下scanf 、fscanf 、sscanf这几个函数的区别,先看一下这几个函数的定义:
scanf函数:
fscanf函数:
sscanf函数:
根据之前所学习的fscanf函数的几个参数,我们可以知道:
scanf:将数据格式化的输出在标准输入流;
fscanf:将数据格式化的输出在任意指定的输出流;
sscanf:将字符串按照一定格式转换为格式化数据;
这里就不在对scanf和fscanf函数进行解释了(上文中已解释),只对sscanf函数进行解释:
我们直接使用案例讲解:
#include<stdio.h> struct Student { char name[10]; int age; }; int main() { struct Student stu = { "zhangsan",20 }; struct Student stu2 = { 0 }; char str[20] = { 0 }; //将stu结构体中的数据格式化转换为字符串后储存到str数组中 sprintf(str, "%s %d", stu.name, stu.age); //将str中的数据格式化输出到stu2结构体中 sscanf(str, "%s %d", stu2.name, &(stu2.age)); printf("%s %d", stu2.name, stu2.age); return 0; }
这样我们就完成了将字符数组中的字符串格式化输出了。
在之前我们知道文件可分为文本文件和二进制文件,那么之前我们学习的都是将数据存储到文本文件中,那么如果我们想存储二进制数据,那该怎么办呢?接下来我们就讲解储存二进制数据的方法。
(3)fwrite函数
先看一下官网对其的解释:
其中的各个参数分别为ptr是一个指向数据存储位置的指针;size是每个数据项的大小;count是要写入的数据项数量;stream是要写入的文件指针,该函数的返回值为实际写入的数据项数量。
现在我们直接使用实例让你更好的理解:
#include<stdio.h> int main() { FILE* pf = fopen("test.txt", "wb"); if (pf == NULL) { perror("fopen"); return 1; } int arr[5] = { 1,2,3,4,5 }; //将arr中的数据以二进制的形式写入文件test.txt中 fwrite(arr, sizeof(int), 5, pf); fclose(pf); pf = NULL; return 0; }
注:由于是二进制写入,所以我们会看不懂其中的内容。实际上fwrite函数的作用就是将指定位置的数据写入到文件中。
(4)fread函数
先看一下官网对其的解释:
其中的各个参数分别为ptr是一个指向数据存储位置的指针;size是每个数据项的大小;count是要读取的数据项数量;stream是要读取的文件指针,该函数的返回值为实际读取的数据项数量。
现在我们继续使用上面fread函数实例让你更好的理解:
#include<stdio.h> int main() { FILE* pf = fopen("test.txt", "rb"); if (pf == NULL) { perror("fopen"); return 1; } int arr[5] = { 0 }; //将文件test.txt中的二进制的数据读取到arr数组中 fread(arr, sizeof(int), 5, pf); //打印读取到的数据 for (int i = 0; i < 5; i++) { printf("%d ", arr[i]); } fclose(pf); pf = NULL; return 0; }
注:由于是二进制读取,所以可以从文件中读取二进制数据,实际上fread函数的作用就是从文件中读取指定数量的数据项,存储到指定位置。
以上就是文件顺序读写的所有方法了!!!
2.文件的随机读写
了解了文件的顺序读写之后,那么我们就产生了一个问题——那么读写到文件的数据能不能不按照顺序呢?其实是可以的。
这里我们就要讲解文件的随机读写的函数:
大致为以下三个,分别为fseek、 ftell、 rewind:
(1)fseek函数
fseek函数用于将文件指针定位到指定位置,它的声明如下:
其中的参数分别为stream是要定位的文件指针,offset是相对于origin的偏移量,origin是定位的起始位置,可以是以下值之一:
SEEK_SET:从文件开头开始计算偏移量;
SEEK_CUR:从当前位置开始计算偏移量;
SEEK_END:从文件末尾开始计算偏移量;
该函数的返回值为一个整数,返回0表示定位成功,返回非零值表示定位失败。
我们直接使用实例来讲解:
#include<stdio.h> int main() { FILE* pf = fopen("test.txt", "r"); if (pf == NULL) { perror("fopen"); return 1; } //将文件指针从开头偏移6个单位 fseek(pf, 6, SEEK_SET); //获取一个字符 int ch = fgetc(pf); printf("%c", ch); fclose(pf); pf = NULL; return 0; }
所以一句话总结fseek函数的作用就是移动文件指针的位置,使你能更灵活的读写文件。
(2)ftell函数
ftell函数用于获取文件指针当前的位置。它的声明如下:
该函数的参数stream是要获取位置的文件指针,函数的返回值为当前文件指针的位置到开始位置的偏移量,以字节为单位,并且如果获取位置失败,ftell函数会返回-1。
我们直接使用实例来讲解:
#include<stdio.h> int main() { FILE* pf = fopen("test.txt", "r"); if (pf == NULL) { perror("fopen"); return 1; } //将文件指针从开头偏移6个单位 fseek(pf, 6, SEEK_SET); //打印偏移量 int ret = ftell(pf); printf("%d", ret); fclose(pf); pf = NULL; return 0; }
这样我们就了解了ftell函数的使用方式。
(3)rewind函数
rewind函数用于将文件指针定位到开始位置。它的声明如下:
该函数的参数stream是要获取位置的文件指针,该函数就是将文件指针定位到开始位置,该函数无返回值。
我们直接使用实例来讲解:
#include<stdio.h> int main() { FILE* pf = fopen("test.txt", "r"); if (pf == NULL) { perror("fopen"); return 1; } //将文件指针从开头偏移6个单位 fseek(pf, 6, SEEK_SET); //打印偏移量 int ret = ftell(pf); printf("%d\n", ret); //将文件指针定位到开始位置 rewind(pf); ret = ftell(pf); printf("%d", ret); fclose(pf); pf = NULL; return 0; }
这样我们就了解了rewind函数的使用方式。
3.文件读取结束的判定
我们知道使文件读取结束有两种可能:
(1)文件在读取的过程中出现异常;
(2)文件读取到了文件的末尾;
那么我们如何去判断文件读取结束时是因为出现异常而停止的还是因为读取到文件的末尾而停止的呢?
这里我们要介绍两个函数:feof和ferror
(1)feof函数
该函数的参数是要判断的文件的文件指针,当是由于读取到文件末尾而停止则返回非0整数,反之为0。
feof 的作用是:当文件读取结束的时候,判断是读取结束的原因是否是:遇到文件尾结束。
(2)ferror函数
该函数的参数是要判断的文件的文件指针,当是由于读取到文件末尾而停止则返回非0整数,反之为0
ferror的作用是:当文件读取结束的时候,判断是读取结束的原因是否是:发生异常导致读取结束。
这里我们直接使用案例来加深理解:
#include <stdio.h> int main() { FILE* pf = fopen("test.txt", "r"); if (pf == NULL) { perror("fopen"); return 1; } //fgetc 当读取失败的时候或者遇到文件结束的时候,都会返回EOF int ch = 0; while ((ch = fgetc(pf)) != EOF) { printf("%c", ch); } //判断是什么原因结束的 if (ferror(pf)) printf("异常导致"); else if (feof(pf)) printf("读取结束导致"); fclose(pf); pf = NULL; return 0; }
这样我们就了解了feof和ferror函数的使用方法。