标准 I/O 库(stdio)及其头文件 stdio.h 为底层 I/O 系统调用提供了一个通用的接口。这个库现在已经成为 ANSI 标准 C 的一部分。标准 I/O 库提供了许多复杂的函数用于格式化输出和扫描输入。在很多方面,你使用的标准 I/O 库的方式和使用底层文件描述符一样,需要先打开一个文件以建立一个访问路径,这个操作的返回值将作为其他 I/O 库函数的参数。在标准 I/O 库中,与底层文件描述符对应的是流(stream,需要注意的是这个流与 C++ 中的输入输出流不一样),它被实现为指向结构 FILE 的指针(文件指针)。
在启动程序时,有 3 个文件流是自动打开的,它们是 stdin、stdout 和 stderr,在 stdio.h 中定义,分别代表着标准输入、标准输出和标准错误输出,与底层文件描述符 0、1、2 相对应。可用的文件流数量与文件描述符一样,都是有限制的,实际的限制由头文件 stdio.h 中定义的 FOPEN_MAX 来定义,它的值至少为 8,在 Linux 系统中,通常是 16。
一、fopen 函数
用于文件和终端的输入输出,,类似于系统调用 open。调用 fopen 函数成功时,返回一个非空的 FILE * 指针,调用失败时,返回 NULL。函数原型如下:
#include <stdio.h>
FILE *fopen(const char *filename,const char *mode);
关于 FILE 结构,有:
typedef struct _iobuf{
int cnt; // 剩余字符数
char *ptr; // 下一个字符的位置
char *base; // 缓冲区的位置
int flag; // 文件访问模式
int fd; // 文件描述符
}FILE;
fopen 打开由 filename 参数指定的文件,并把它与一个文件流关联起来。mode 参数指定文件打开的方式,可以取以下的一些值:
字符串 说明"r" 、"rb" 以只读方式打开文件
"w" 、"wb" 以写方式打开文件,并把文件长度截短为零
"a" 、"ab" 以写方式打开文件,新内容追加在文件尾
"r+" 、"rb+" 、"r+b" 以更新方式打开文件(读和写)
"w+" 、"wb+" 、"w+b" 以更新方式打开文件,并把文件长度截短为零
"a+" 、"ab+" 、"a+b" 以更新方式打开文件,新内容追加在文件尾
其中,字母 b 表示文件是一个二进制文件而不是文本文件。另外,需要注意的是,mode 参数是一个字符串而不是一个字符,因此总是需要使用双引号 " " 括起来。
二、fread 函数
fread 函数用于从一个文件流里读取数据。调用 fread 成功时,返回成功读到缓冲区里的记录个数(不是字节数)。函数原型如下:
#include <stdio.h>
size_t fread(void *ptr,size_t size,size_t nitems,FILE *stream);
数据从文件流 stream 读到由 ptr 指向的数据缓冲区里,size 参数指定每个数据记录的长度,计数器 nitems 给出要传输的记录个数。举个例子:
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
int main()
{
int buffer[10];
int i;
int counter;
counter = fread(buffer,1,10,stdin);
printf("读取到缓冲区的记录个数为:%d\n",counter);
fwrite(buffer,sizeof(int),1,stdout);
exit(0);
}
这段代码从标准输入读取 10 个 字符到缓冲区中,然后调用 fwrite 函数将缓冲区中的前 4 个字符写到标准输出。
三、fwrite 函数
fwrite 函数从指定的数据缓冲区里取出数据记录,并把它们写到输出流中,返回成功写入的记录个数,其参数与 fread 函数的参数类似。
#include <stdio.h>
siize_t fwrite(const void *ptr,size_t size,size_t nitems,FILE *stream);
需要注意的是,用 fwrite 写的文件在不同的计算机体系结构之间可能不具备可移植性,因此,不推荐把 fread 和 fwrite 用于结构化数据。演示代码如下:
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
int main()
{
char buffer[] = "hello,my name is Linux公社";
fwrite(buffer,1,sizeof(buffer),stdout);
exit(0);
}
测试下
[linuxidc@localhost linuxidc.com]$ ./linuxidc
输出:
hello,my name is Linux公社
这段代码将 buffer 中的内容写入标准输出。
四、fclose 函数