示例:
将内存中的数据以二进制的方式输出到文件
typedef struct student { char name[10]; int age; float stature; }student; #include <stdio.h> int main() { FILE* pFile; student s1 = { "初阶牛",20,1.755 }; pFile = fopen("123.txt", "wb"); if (pFile != NULL) { fwrite(&s1, sizeof(s1), 1, pFile); } else { printf("打开失败"); return 1;//返回非0 } //关闭文件 fclose(pFile); return 0; }
运行结果:
由于"初阶牛"是字符型,二进制显示也是这样,而其他的数据就显示出来我们就看不懂了.
函数模型:
参数说明:
参数 | 含义 |
ptr | 指向大小至少为 (size*count) 字节的内存块的指针,用于存放待会要从流读取到的数据 |
size | 要读取的每个元素的大小(以字节为单位)。 |
count | 元素个数 |
stream | 指向指定输入流的 FILE 对象的指针。 |
图解:
示例:
将文件中的数据以二进制的方式读取到内存
typedef struct student { char name[10]; int age; float stature; }student; #include <stdio.h> int main() { FILE* pFile; student s1 ; pFile = fopen("123.txt", "rb");//此时里面已经有了二进制数据 if (pFile != NULL) { fread(&s1, sizeof(s1), 1, pFile); } else { printf("打开失败"); return 1;//返回非0 } //关闭文件 fclose(pFile); printf("%s %d %.2f", s1.name, s1.age, s1.stature); return 0; }
4.2 文件的随机读写
fseek函数:
函数模型:
参数介绍:
参数 | 含义 |
stream | 指向标识流的 FILE 对象的指针。 |
offset | 二进制文件:要从源偏移的字节数。文本文件:零或 ftell 返回的值。 |
origin | 用作偏移参考的位置。 |
origin :参考位置表
示例:文件中已经有了,数据:Hello CSDN!!!
#include <stdio.h> int main() { FILE* pFile; pFile = fopen("test6.txt", "r");//文件中已经 有了数据:Helllo CSDN!!! if (pFile == NULL) { perror(fopen); } else { char tmp= fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); printf("\n"); //调整偏移量 fseek(pFile,-3, SEEK_CUR);//将光标从文件当前处,往前偏移3个位 tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); printf("\n"); fseek(pFile, 1, SEEK_SET);//将光标从文件开头处,往后偏移一个位 tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); printf("\n"); fseek(pFile, -8, SEEK_END);//将光标从文件结尾处,往前偏移8个位 tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); } fclose(pFile); return 0; }
运行结果:
Hello
ll
el
CSDN!
解释:
ftell函数
函数模型:
参数介绍:
参数 | 含义 |
stream | 指向标识流的 FILE 对象的指针。 |
函数功能,获取流中的当前位置的偏移量.
示例:
#include <stdio.h> int main() { FILE* pFile; pFile = fopen("test6.txt", "r");//文件中已经 有了数据:Helllo CSDN!!! if (pFile == NULL) { perror(fopen); } else { char tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); tmp = fgetc(pFile); printf("%c", tmp); printf("\n"); printf("%d", ftell(pFile)); } fclose(pFile); return 0; }
结果:
Hello
5
rewind函数
函数模型:
函数功能:
将流的位置设置为开头.
与fseek(pFile, 0, SEEK_SET)功能一样,就不过多介绍了.
4.3 文本文件 与 二进制文件的区别
数据存储的形式有多种,数据文件被称为文本文件或者二进制文件。
数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。
如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件就是文本文件。
那么一个数据在内存中是怎么存储的呢?
字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。
示例:
以整形数字:520为例
#include <stdio.h> int main() { int a = 520; FILE* pf = fopen("test.txt", "wb"); fwrite(&a, sizeof(int), 1, pf);//二进制的形式写到文件中 fclose(pf); pf = NULL; return 0; }
结果:
我们可以用vs,右击添"现有项",将文件添加进来,然后打开方式选择二进制编译器
这是小端存储模式,所以16进制应该为00 00 02 08,这便是520转化为16进制的值.
用文本文件的方式去写
#include <stdio.h> int main() { int a = 520; FILE* pf = fopen("test.txt", "wb"); fprintf(pf,"%d",a);//文本的形式写到文件中 fclose(pf); pf = NULL; return 0; }
运行结果:
一般以文本文件的方式去写,会占用更多的字节空间,因为对于每一位数字都要单独转化为ASCII码值.
例如:
文本520,用ASCII码值(16进制)表示为35 32 30,占3个字节
二进制520,用ASCII码值(16进制)表示为00 00 02 08,占四个字节.
啊哦,这里的例子不大合适,如果数字是一个大于4位的数字,比如5201314,那么
文本文件:占8个字节
二进制文件:占4个字节.
五、文件结束的判定
feof函数
注意:
feof函数经常被错用为是判断文件是否结束.而在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束。
而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。
- 文本文件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets )
例如:
fgetc 判断是否为 EOF .
fgets 判断返回值是否为 NULL .
- 二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。
例如:
fread判断返回值是否小于实际要读的个数。
六、文件缓冲区
ANSIC 标准采用“缓冲文件系统”处理的数据文件的,所谓缓冲文件系统是指系统自动地在内存中为程序
中每一个正在使用的文件开辟一块“文件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才一起送到磁盘上。如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的大小根据C编译系统决定的.
因为有缓冲区的存在,C语言在操作文件的时候,需要做刷新缓冲区(flush函数)或者在文件操作结束的时候关闭文件。
如果不刷新,可能导致读写文件的问题