行I/O其实可以用两种方式执行——未格式化的或者格式化的。这两种形式都用于操纵字符串。区别在于未格式化的I/O只是通过fgets和fputs简单读取或写入字符串,而格式化的I/O则执行数字和其他变量的内部或外部表示形式之间的转换。由于日常工作中操作的一般都是格式化I/O,因此这里不讲fgets和fputs这种非格式化I/O操作了。(当然,还有一个重要的原因,fgets无法判断缓冲区长度,容易导致溢出等情况)
int fscanf(FILE* stream, const char* format, ...);
int scanf(const char* format, ...);
int sscanf(const char* string, const char* format, ...);
这些函数都从输入源读取字符并根据format字符串给出的格式化代码对它们进行转换。当格式化字符串到达末尾或者读取的输入不再匹配格式字符串所指定的类型时,输入就停止。在任何一种情况下,被转换的输入值的数目作为函数的返回值返回。如果在任何输入值被转换之前文件就已经到达尾部,函数就返回常量值EOF。
printf家族
int fprintf(FILE *stream, const char* format, ...);
int printf(const char* format, ...);
int sprintf(char* buffer, const char* format, ...);
二进制I/O
把数据写到文件里效率最高的方法是用二进制形式写入,而且Android系统里也有很有用二进制文件通过位来存储数据的应用场景。介绍一下操纵二进制I/O的函数原型。
fread函数用于读取二进制数据,fwrite函数用于写入二进制数据。它们的原型如下所示:
size_t fread(void* buffer, size_t size, size_t count, FILE* stream);
size_t fwrite(void* buffer, size_t size, size_t count, FILE* stream);
buffer是一个指向用于保存数据的内存位置的指针,size是缓冲区中每个元素的字节数,count是读取或写入的元素数,stream是数据读取或写入的流。
刷新和定位函数
在处理流时,另外还有一些函数也较为有用。首先,是fflush,它迫使一个输出流的缓冲区内的数据进行物理写入,不管它是不是已经写满。它的原型如下所示:
int fflush(FILE* stream);
当我们需要立即把输出缓冲区的数据进行物理写入时,应该使用这个函数。
在正常的情况下,数据以线性的方式写入,这意味着后面写入的数据在文件中的位置是在以前所有写入数据的后面。C同时支持随机访问I/O,也就是以任意顺序访问文件的不同位置。随机访问是通过在读取或写入前先定位到文件中需要的位置来实现的。一般使用fseek函数来实现,函数原型如下:
int fseek(FILE* stream, long offset, int from);
fseek函数允许你在一个流中定位。这个操作将改变下一个读取或写入的位置。它的第一个参数是需要改变的流,它的第二个和第三个参数标识文件中需要定位的位置。下表描述了fseek参数的使用方法。
如果from是 |
你将定位到 |
SEEK_SET |
从流的起始位置起offset个字符,offset必须是一个非负值 |
SEEK_CUR |
从流的当前位置起offset个字符,offset可正可负 |
SEEK_END |
从流的尾部位置起offset个字符,offset值可正可负。为正数时,它将定位到文件尾的后面 |