APUE读书笔记-05标准输入输出库(1)

1、简介

在本章,我们描述了标准输入输出库。由于这个库也被许多非UNIX的操作系统实现了,所以这个库由 ISO C 标准指定。其它的接口被 SUS 定义,作为 ISO C 标准的扩展。

标准输入输出库处理了诸如缓存分配,以及分配优化的大小的块进行 I/O ,这样我们就不必担心如何使用正确的块大小进行 I/O 了(如原书3章9节所描述的那样)。这样,这个库就很容易使用了,但是同时,也引入了我们没有预料到的其他一些问题。

标准 I/O 库由Dennis Ritchie在大约1975年编写,它作为Mike Lesk编写的可移植 I/O 库的主要版本.并且在30年的期间内,对它改动非常少。

译者注

原文参考

参考: APUE2/ch05lev1sec1.html

2、流和文件描述符号

我们使用系统调用来操作文件的时候,是通过文件描述符号来标识这个文件,如果我们使用标准库函数来进行文件操作的我们通常是把一个 stream 和这个文件相关联了,然后通过这个 stream 对这个流进行操作,也即,标准 I/O 库是以流( stream )为中心进行操作的。

当我们使用库函数 fopen 打开一个文件的时候,会返回一个指向 FILE 对象的指针,这个对象包含了 I/O 库管理 stream 需要的所有信息,例如被操作的文件的描述符号,指向这个 stream 的缓存的指针,以及缓存的大小(注意这个缓存和系统调用时可能经过的内核的缓存不同,经过内核的缓存更接近下层,在第3章8节讨论 I/O 效率时的最后也提到过,但这属于内核驱动范围了,可以暂时忽略),当前缓存中的字符数目,错误标志等等。我们把这个 FILE 对象做为参数传递给标准输入输出库函数,就可以实现相应的操作了(使用库函数操作,也不用费心去关心输入输出缓存设置为多大才好了)。

应用程序不用检验 FILE 对象。引用一个流,只需将 FILE 指针作为参数传递给每个标准 I/O 函数。本书中,我们称指向 FILE 的指针(类型为 FILE* )为文件指针。

通常,有单字节和多字节字符。 streamorientation 用于决定读写的是单字节字符还是多字节字符。在最开始一个 stream 创建的时候,它是没有 orientation 的,这时候,如果使用了一个多字节 I/O 函数对这个 stream 进行操作(例如 <wchar.h> 中的函数),那么这个 stream 就被设置为 wide-oriented 的;同理如果用单字节的 I/O 函数对这个 stream 进行操作,那么这个 stream 就被设置为 byte-oriented 的。一旦设置了 stream 的方向,除非被关闭,否则就不能改变了。当一个 streamset 的时候,只有两个函数可以改变这个 streamorientation 。它们是: fwidefreopen .

freopen 原型如下:

FILE *freopen(const char *path, const char *mode, FILE *stream);

它会把 streamorientationclear .把原来的 stream 给关掉,然后把参数指定的新的 stream 和这个文件相互关联。

fwide 原型如下:

#include <stdio.h>
#include <wchar.h>
int fwide(FILE *fp, int mode);

这里,如果 mode 是负数,那么设置成 byte-oriented ;如 mode 是正数,那么设置成 wide-oriented ;如果 mode 是0那么不会做设置,但也返回当前流的 orientation 值。

需要注意的是 fwide 不会改变已经被 orientedstream ,也不会返回 error (也就是那个流刚开始被创建之后没有被使用过就没有被设置过,这个时候则可以用这个函数来设置)。

译者注

原文参考

参考: APUE2/ch05lev1sec2.html

3、标准输入、标准输出和标准错误

有三个重要的流: stdin , stdout , stderr 表示标准输入,输出和错误流,它们对应的文件描述符号是: STDIN_FILENO , STDOUT_FILENO ,和 STDERR_FILENO .

译者注

原文参考

参考: APUE2/ch05lev1sec3.html

4、关于库函数和 buffer

使用标准输入输出( Standard I/O )库函数进行操作的目的就是为了使用最小数目的的 readwrite 系统调用(通过自动管理设置多大的缓存等等)。 Standard I/O 一般选择合适的大小来分配缓存。可能是 BUFSIZ 常量的值(在 <stdio.h> 中定义),也可能是 stat 结构中的 st_blksize 成员。

(1)三种类型的缓存

  • Fully Buffered .在这种情况下当 stand i/o 的缓存被填满的时候才会发生 i/o .一般使用 standard i/o 操作磁盘上面的文件的时候使用的是这种类型的 buffer .

    这里需要注意在UNIX环境中,刷新( flush )这个术语有两种意思。在标准 I/O 库方面,刷新意味着将缓存中的内容写到磁盘上(该缓存可以只是局部填写的)。在终端驱动程序方面(例如在第11章中所述的 tcflush 函数),刷新表示丢弃已存在缓存中的数据。这里的刷新( flush )是指标准 I/O 缓存的写操作。缓存可由标准 I/O 例程自动地刷新(例如当填满一个缓存时) ,或者可以调用函数 fflush 刷新一个流。

  • Line Buffered .在这种情况下,当输入或者输出的时候遇到了一个换行符号的时候才会发生 i/o .一般终端的标准输入和标准输出会用到这种类型的缓存。

    对于这种 Line Buffered ,需要注意的是: Line Buffer 的缓存大小是固定的,当遇到换行符号之前缓存被填满了的时候,也会发生 i/o 。还有一个就是:当从一个 unbuffer 的方式进行请求输入的时候;以及当从 Line Buffered 请求输入的时候;这两种情况下也都会刷新( flushed )行缓存输出的缓存。

  • Unbuffered . standard i/o 不会缓存字符。一般标准错误流会使用这种类型的缓存,这样可以保证信息尽可能快地出现。

ISO C 规定:

  • standard input /output = 当引用的不是交互式的设备的时候,是 =fully buffered .
  • standard error 一定不能是 fully buffered .

但是,上面的规定却不是非常确定的,一般而言:

  • standard errorunbuffered .
  • 其他的 stream 如果它们引用的是 terminal 设备,那么就是 line Buffered ;否则是 fully buffered .

(2)设置缓存

如果不是用默认的缓存类型,有两个可以设置缓存的函数:

#include <stdio.h>
void setbuf(FILE *restrict fp, char *restrict buf);
int setvbuf(FILE *restrict fp, char *restrict buf, int mode, size_t size);

这两个函数必须在 streamopen 之后,并且任何 i/o 操作之前,被调用。

使用 setbuf 我们可以设置和取消 buffer .如果设置 buffer ,指向 buf 的指针的数据的大小必须是 BUFSIZ ,一般这样 stream 就是 fully buffered .但是如果 stream 是和 terminal 相关联的也有些系统会把 buffered 设置为 line buffered 的。如果取消 buffer ,那么 buf 参数是 NULL 的。

使用 setvbuf ,我们可以精确地指定我们想要的 buffer 类型。我们通过 modde 参数来进行指定:

  • _IOFBF: 表示 fully buffered .
  • _IOLBF: 表示 line buffered .
  • _IONBF: 表示 unbuffered .

如果我们指定为 unbuffered ,那么 bufsize 参数会被忽略.

如果指定了 fully buffered 或者 line buffered 那么:我们可以通过 bufsize 来设置缓存和大小。如果这时候的 buf 参数是 NULL 那么 standio = 会自动分配一个合适的大小,例如 =BUFSIZ .

注意,你需要保证在 stream closed 的时候, buf 是存在的。如果我们自行指定分配的 buffer 是一个函数内部的局部变量,我们需要在这个函数返回之前把这个 streamclosed 了,另外,因为某些实现使用这个 buffer 的一部分做为记录的空间,所以我们存放在这个 buffer 中的数据应该小于它的实际大小。一般我们应该让系统自己选择分配合适的缓存和大小。

有些C库使用 stat 结构的 st_blksize 作为最优输入输出缓存大小,通过后面的讨论我们能够看出, GNU C 库就是使用这个方法。

                                关于setbuf和setvbuf函数
+--------------------------------------------------------------------------------------------------------+
| Function |  mode  |    buf    |          Buffer and length           |        Type of buffering        |
|----------+--------+-----------+--------------------------------------+---------------------------------|
|          |        | non-null  | user buf of length BUFSIZ            | fully buffered or line buffered |
|setbuf    |        |-----------+--------------------------------------+---------------------------------|
|          |        | NULL      | (no buffer)                          | unbuffered                      |
|----------+--------+-----------+--------------------------------------+---------------------------------|
|          |        | non-null  | user buf of length size              |                                 |
|          |_IOLBF  |-----------+--------------------------------------|fully buffered                   |
|          |        | NULL      | system buffer of appropriate length  |                                 |
|          |--------+-----------+--------------------------------------+---------------------------------|
| setvbuf  |        | non-null  | user buf of length size              |                                 |
|          |_IOFBF  |-----------+--------------------------------------|line buffered                    |
|          |        | NULL      | system buffer of appropriate length  |                                 |
|          |--------+-----------+--------------------------------------+---------------------------------|
|          | _IONBF | (ignored) | (no buffer)                          | unbuffered                      |
+--------------------------------------------------------------------------------------------------------+

我们需要注意,如果我们以一个函数中的自动变量为标准输入输出分配缓存,那么我们需要在返回函数之前关闭流。另外,有些实现使用这个缓存的部分内容作为内部索引之用,所以实际存放于缓存中的数据会比缓存的大小要小。一般来说,我们让系统自己选择缓存大小,以及自动分配缓存。这样,当我们关闭流的时候,标准 I/O 库会自动释放缓存。

我们可以用 fflush 函数将流刷新,其声明如下:

#include <stdio.h>
int fflush(FILE *fp);

返回:如果成功返回0,如果错误返回 EOF

fflush 会导致在 stream 中没有被 written 的数据被提交到 kernel 中去。特别地,如果 fpNULL ,那么这个函数会导致所有的输出流被刷新。

译者注

原文参考

参考: APUE2/ch05lev1sec4.html

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容