对于标准 I/O 库,它们的操作是围绕流(stream)进行的。当用标准 I/O 库打开或创建一个文件时,我们已使一个流与一个文件相关联。
#include
#include
int fwide(FILE *fp, int mode);
/* 返回值:
若流是宽定向的,返回正值
若流是字节定向的,返回负值
若流是未定向的,返回0
*/
流的定向:决定了所读、写的字符是单字节或多字节(“宽”)的。当一个流最初被创建时,它并没有定向。若在未定向的流上使用一个多字节的 I/O 函数,则将该流的定向设置为宽定向的。若在未定向的流上使用一个单字节的 I/O 函数,则将该流的定向设为字节定向的。
标准 I/O 库提供缓冲的目的是尽可能减少使用 read 和 write 调用的次数。一共提供了以下三种缓冲
全缓冲:在这种情况下,在填满标准 I/O 缓冲区后才进行实际 I/O 操作。
行缓冲:当在输入和输出中遇到换行符时,标准 I/O 库执行 I/O 操作。
不带缓冲:标准 I/O 库不对字符进行缓冲存储。
标准错误流 stderr 通常时不带缓冲的,这就使得出错信息可以尽快显示出来,而不管它们是否含有一个换行符。
很多系统默认使用下列类型的缓冲:
#include
void setbuf(FILE *restrict fp, char *restrict buf);
int setvbuf(FILE *restrcit fp, char *restrict buf, int mode, size_t size);
// 返回值:若成功,返回 0;若出错,返回非 0
因为每个函数都要求一个有效的文件指针作为它们的第一个参数,所以这些函数一定要在流已被打开后调用。
setbuf 函数打开或关闭缓冲机制:为了带缓冲进行 I/O,参数 buf 必须指向一个长度为 BUFSIZ 的缓冲区(该常量定义在
setvbuf 可以精确的说明所需的缓冲类型,用 mode 参数实现:
_IOFBF 全缓冲
_IOLBF 行缓冲
_IONBF 不带缓冲
如果指定一个不带缓冲的流,则忽略 buf 和size 参数。如果指定全缓冲或行缓冲,则 buf 和 size 可选择地指定一个缓冲区及其长度。如果该流本身是带缓冲的,而 buf 是 NULL,则标准 I/O 库将自动地为该流分配适当长度的缓冲区(即 常量 BUFFSIZ 所指的值)。
在 UNIX 环境中,flush 有两种意思,在表中 I/O 库方面,flush(冲洗)意味着将缓冲区中的内容写到磁盘上(该缓冲区可能至少部分填满的)。在终端驱动程序方面,flush(刷清)表示丢弃已存储在缓冲区中的数据。
任何时候,都可以使用 fflush 强制冲洗一个流。
#include
int fflush(FILE *fp);
// 返回值:若成功,返回0;若出错,返回 EOF
此函数是该流所有未写的数据都被传送至内核。作为一种特殊情形,如若 fp 是 NULL,则此函数将导致所有输出流被冲洗。
#include
FILE *fopen(const char * restrict pathname, const char *restrict type);
FILE *freopen(const char *restrict pathname, const char *restrict type, FILE *restrict fp);
FILE *fdopen(int fd, chonst char *type);
// 返回值:若成功,返回文件指针,若出错,返回 NULL
这 3 个函数的区别如下:
type参数指定对该 I/O流的读、写方式
r 或 rb 为读而打开
w 或 wb 为写而创建或把文件截断至 0 长
a 或 ab 追加:为在文件尾写而打开,或为写而创建
r+ 或 r+b 或 rb+ 为读和写而打开
w+ 或 w+b 或 wb+ 为读和写而打开或把文件截断至 0 长
a+ 或 a+b 或 ab+ 为在文件尾读和写而打开或创建
使用字符 b 作为 type 的一部分,这使得标准版 I/O 系统可以区分文本文件和二进制文件。
对于 fdopen 为写而打开并不截断该文件。
注:在指定 w 或 a 类型创建一个新文件时,我们无法说明该文件的访问权限位。可以通过挑中 umask 值来限制这些权限。 此外,除非流引用终端设备,否则按系统默认,流被打开时是全缓冲的。若流引用终端设备,则该流是行缓冲的。
#include
int fclose(FILE * fp);
// 返回值:若成功,返回 0;若出错,返回 EOF
在该文件关闭之前,冲洗缓冲区的输出数据。缓冲区的任何输入数据被丢弃。如果标准 I/O 库已经为该流自动分配了一个缓冲区,则释放此缓冲区。