**From : https://www.gnu.org/software/libc/manual/html_node/Stream-Buffering.html**

12.20 流缓冲

通常情况下,写入流的字符会在写入前进行累积然后以块的形式异步转送而非由应用程序马上输出。相似的,流通常从主机环境以块的形式而非字节-字节的形式读入数据。这称为缓冲。

如果你正在写一个用流来交互的程序,当你设计交互接口时你需要理解缓冲是如何工作的。否则,你可能会发现输出(例如进程提示信息)不输出理想值,或者出现奇怪的行为。

这一节讲解的是在流/文件/设备之间传输设备,并不会涉及echoing, flow control, 特殊设备。如果想了解关于终端设备的通用控制操作符的信息,参考Low-Level Terminal Interface

你可以通过使用底层I/O函数与文件描述符来来避免使用流缓冲。参考 Low-Level I/O

12.20.1 概念与术语

一共有三种缓冲策略:(译者注:缓冲策略是写入流/文件的充分条件,不是必要的。缓冲区存在的意义就是在使用“Stream-level I/O”时从缓冲区进行异步块写入/读出,这样可以在设备堵塞的时候加快效率。如果一次只写入少量数据,内核一看没有堵塞,“干脆”就把缓冲区的内容写入了,反正放着也是放着)

  • 无缓冲 unbuffered :从一个无缓冲的流中读写会马上产生效果
  • 行缓冲 line buffered:当遇到一个换行符的时候字符会以块的形式读写。
  • 满缓冲 fully buffered:字符会以任意大小的块写入读出。(真的是直译。。感觉和网上一些说满的时候才读写的说法不一样,说明可能是不堵塞的时候就读写缓冲区,最多等到缓冲区满

新开的流一般是满缓冲的,只有一个例外:当流是一个可交互设备(例如终端)的时候,流将变为行缓冲。如果想了解关于如何选择缓冲策略,参考 Controlling Buffering 。通常情况下,默认会选择出最方便的缓冲策略。

在行缓冲下,以换行符结束的信息会马上输出到交互设备里——这通常是你想要的。不以换行符结尾的信息可能不会马上显示到交互设备,所以如果你想要立即显示,你需要在写入后使用fflush, 参考 Flushing Buffers.(译者注:通常使用fprintf+stderr,因为stderr默认是无缓冲的)

12.20.2 清除缓冲区

清除缓冲区意味着立即以块的形式写入缓冲区内收集的内容。有很多情况下缓冲区会自动清除:

  • 当输出缓冲区满后尝试输出。
  • 当流关闭的时候。参考 Closing Streams
  • 当程序通过调用exit结束的时候。参考 Normal Termination
  • 当向行缓冲缓冲区写入换行符时。
  • 当从本文件的任意流读入时。

If you want to flush the buffered output at another time, call fflush, which is declared in the header file stdio.h.如果你想要在别的时候清除缓冲区,可以使用stdio.h中声明的fflush

  • Function: int fflush (FILE *stream)

Preliminary: | MT-Safe | AS-Unsafe corrupt | AC-Unsafe lock corrupt | See POSIX Safety Concepts.这个函数会导致与stream关联的缓冲区被写入到文件中,如果stream是一个空指针,那么fflush会导致目前打开的所有流的缓冲区被清除。这个函数将返回EOF如果发生一个写入错误,否则返回0。

  • Function: int fflush_unlocked (FILE *stream)

Preliminary: | MT-Safe race:stream | AS-Unsafe corrupt | AC-Unsafe corrupt | See POSIX Safety Concepts. fflush_unlocked函数和fflush相同,除了它不会隐式的锁上这个stream流。

此处省略一些...

  • Function: void **_flushlbf** (void)

Preliminary: | MT-Safe | AS-Unsafe corrupt | AC-Unsafe lock corrupt | See POSIX Safety Concepts.该函数会清除所有的行缓冲流的缓冲区,声明在stdio_ext.h头文件中。

兼容性: 有一些脑子坏掉的操作系统总是对那些以换行符为导向的输入“念念不忘”——清除一个行缓冲会导致一个换行符被写入。幸运的是,这个特性越来越少见,并且对于GNU C Library 你不用担心。

有一些情况下不去手动清除缓冲区而是忘记这件事可能是一个更好的选择,因为读写可能不是必要的而且是相对花费大的。

  • Function: void **__fpurge** (FILE *stream)

Preliminary: | MT-Safe race:stream | AS-Unsafe corrupt | AC-Unsafe corrupt | See POSIX Safety Concepts.__fpurge 函数会把stream流的缓冲区清空,但是不会产生读写!声明在stdio_ext.h。

12.20.3 控制缓冲策略

在打开一个流之后,你通过使用setvbuf选择该流使用何种缓冲策略。

以下列出的函数和宏在头文件stdio.h中声明。

  • Function: int setvbuf (FILE *stream, char buf, int mode, size_t size)*

Preliminary: | MT-Safe | AS-Unsafe corrupt | AC-Unsafe lock corrupt | See POSIX Safety Concepts.这个函数是用来定义stream流该采用何种缓冲策略——可以是 _IOFBF (满缓冲), _IOLBF (行缓冲), or _IONBF (无缓冲)。如果你输入的一个空指针作为buf参数,那么setvbuf会自动使用malloc申请一块内存,当你关闭流的时候,缓冲区会被清除释放掉。否则buf对应的内存块至少应该是size大小。你不应该释放掉buf对应的空间只要流还没有关闭。你应该确保buf对应的内存是静态存储的(例如使用malloc)。使用一个自动存储期限的buf块不是一个好的选择——除非在退出当前块之前关掉了流。当buf对应的数组块是缓冲区的时候,stream流i/o函数会使用这个内存块作为一些内部用途——所以你不应该试着去直接访问这个数组的值当它被使用的时候。 setvbuf 成功时返回0,否则返回非零数当mode是不可取的或者要求不能被满足。

  • Macro: int **_IOFBF**

这个宏的值是一个整数常量表达式,可以被setvbuf函数用来是缓冲区是满缓冲的。

  • Macro: int **_IOLBF**

这个宏的值是一个整数常量表达式,可以被setvbuf函数用来是缓冲区是行缓冲的。

  • Macro: int **_IONBF**

这个宏的值是一个整数常量表达式,可以被setvbuf函数用来是缓冲区是无缓冲的。

  • Macro: int BUFSIZ

这个宏的值是一个整数常量表达式,可以被setvbuf函数用来表达size,这个值被保证最小是256。 BUFSIZ 是由操作系统选择的,以此来提高i/o的效率。所以使用 BUFSIZ 作为setvbuf的大小值是一个很好的选择。事实上,你可以用过fstat 系统调用获得一个更好的值(在文件属性的 st_blksize 区域),参考 Attribute Meanings.有时候人们使用 BUFSIZ作为申请内存空间的大小值(或者作为i/o操作的内存,例如fgets)——这没什么特别的理由,除了能提高一些i/o效率。

  • Function: void setbuf (FILE *stream, char buf)*

Preliminary: | MT-Safe | AS-Unsafe corrupt | AC-Unsafe lock corrupt | See POSIX Safety Concepts.If buf is a null pointer, the effect of this function is equivalent to calling setvbuf with a mode argument of_IONBF. Otherwise, it is equivalent to calling setvbuf with buf, and a mode of _IOFBF and a size argument of BUFSIZ.The setbuf function is provided for compatibility with old code; use setvbuf in all new programs.如果buf是一个空指针,这个函数的效果等于setvbuf使用_IONBF.否则,等效于使用 setvbuf_IOFBF BUFSIZ这两个参数。该函数是为了兼容一些老的代码,在新的程序中请使用setvbuf.

  • Function: void setbuffer (FILE *stream, char buf, size_t size)*

Preliminary: | MT-Safe | AS-Unsafe corrupt | AC-Unsafe lock corrupt | See POSIX Safety Concepts.

省略...这个函数是为了兼容一些BSD的老旧代码。请使用setvbuf.

  • Function: void setlinebuf (FILE *stream)

Preliminary: | MT-Safe | AS-Unsafe corrupt | AC-Unsafe lock corrupt | See POSIX Safety Concepts.

省略...这个函数是为了兼容一些BSD的老旧代码。请使用setvbuf.

剩下一些不常用的函数就不翻译了。

C语言 流缓冲的更多相关文章

  1. C语言 流缓冲 Stream Buffering

    From : https://www.gnu.org/software/libc/manual/html_node/Stream-Buffering.html 译者:李秋豪 12.20 流缓冲 通常情 ...

  2. c++流缓冲学习---rdbuf()

    我们使用STL编程的时候有时候会想到把一个流对象指向的内容用另一个流对象来输出,比如想把一个文件的内容输出到显示器上,我们可以用简单的两行代码就可以完成: ifstream infile(" ...

  3. Java基础知识强化之IO流笔记41:字符流缓冲流之复制文本文件案例02(使用 [ newLine() / readLine() ] )(重要)

    1. 使用字符流缓冲流的特殊功能 [ newLine() / readLine() ] 需求:把当前项目目录下的a.txt内容复制到当前项目目录下的b.txt中  数据源: a.txt -- 读取数据 ...

  4. Java基础知识强化之IO流笔记39:字符流缓冲流之复制文本文件案例01

    1. 字符流缓冲流之复制文本文件案例 需求:把当前项目目录下的a.txt内容复制到当前项目目录下的b.txt中 数据源: a.txt -- 读取数据 -- 字符转换流 -- InputStreamRe ...

  5. Java基础知识强化之IO流笔记38:字符流缓冲流之BufferedWriter / BufferedReader使用

    1. 字符流缓冲流: 字符流为了高效读写,也提供了对应的字符缓冲流. BufferedWriter:字符缓冲输出流 BufferedReader:字符缓冲输入流 2. BufferedWriter使用 ...

  6. (19)IO流之字符流FileReader和FileWriter,缓冲字符流---缓冲输入字符流BufferedReader和缓冲输出字符流BufferedWriter

    字符流,读取的文件是字符的时候,有两个基类一个是Reader,一个是Writer这有点拟人的感觉,人直接看懂的是文字 字符流 字节流:读取的是文件中的二进制字节流并不会帮你转换成看的懂得字符 字符流: ...

  7. -1-4 java io java流 常用流 分类 File类 文件 字节流 字符流 缓冲流 内存操作流 合并序列流

      File类 •文件和目录路径名的抽象表示形式 构造方法 •public File(String pathname) •public File(String parent,Stringchild) ...

  8. IO流(字节流,字符流,缓冲流)

    一:IO流的分类(组织架构) 根据处理数据类型的不同分为:字节流和字符流 根据数据流向不同分为:输入流和输出流   这么庞大的体系里面,常用的就那么几个,我们把它们抽取出来,如下图:   二:字符字节 ...

  9. java基础46 IO流技术(输出字符流/缓冲输出字符流)

    一.输出字符流 1.1.输出字符流体系 --------| Writer:输出字符流的基类(抽象类)  ----------| FileWriter:向文件输出数据输出字符流(把程序中的数据写到硬盘中 ...

随机推荐

  1. webpack配置这一篇就够

    最近看了一篇好文,根据这个文章重新梳理了一遍webpack打包过程,以前的一些问题也都清楚了,在这里分享一下,同时自己也做了一些小的调整 原文链接:http://www.jianshu.com/p/4 ...

  2. Ubuntu 14.02 cmake升级 失败解决

    错误的提示: CMake Error: Could not find CMAKE_ROOT !!! CMake has most likely not been installed correctly ...

  3. 软件工程(GZSD2015)学生博客列表

    2015年贵州师范大学软件工程课程学生博客列表 陈小丽 郑倩 唐洁 周娟 李利思 肖俊 罗文豪 周静 徐明艳 毛涛 邓洪虹 岳庆 李盼 安坤 何亚 涂江凤 张义平 杨明颢 杨家堂 胡贵玲 寿克霞 吴明 ...

  4. 团队作业4——第一次项目冲刺(Alpha版本) 2

    一.Daily Scrum Meeting照片 二.燃尽图 三.项目进展 1.完成了大部分查重算法的书写. 余弦查重算法 2.其他非主页面的部分设计. 查重过程的显示页面 四.困难与问题 1.算法是程 ...

  5. 第二次项目冲刺(Beta阶段)5.24

    1.提供当天站立式会议照片一张 会议内容: ①检查前一天的任务情况. ②制定新一轮的任务计划. 2.每个人的工作 (1)工作安排 队员 今日进展 明日安排 王婧 #63Web输出以文件名为标题 #63 ...

  6. 201521123006 《Java程序设计》第6周学习总结

    1. 本周学习总结 1.1 面向对象学习暂告一段落,请使用思维导图,以封装.继承.多态为核心概念画一张思维导图,对面向对象思想进行一个总结. 注1:关键词与内容不求多,但概念之间的联系要清晰,内容覆盖 ...

  7. 201521123089 《Java程序设计》第4周学习总结

    1. 本周学习总结 1.1 尝试使用思维导图总结有关继承的知识点. 1.2 使用常规方法总结其他上课内容. 总结: (1)类名的首字母一定要大写. (2)制类型转换:类强制转换为子类时只有当引用类型真 ...

  8. 201521123054 《Java程序设计》第11周学习总结

    1. 本周学习总结 2. 书面作业 1.1 除了使用synchronized修饰方法实现互斥同步访问,还有什么办法实现互斥同步访问(请出现相关代码)? 使用Lock对象和Condition对象实现互斥 ...

  9. Java:final、static关键字 详解+两者结合使用

    一  final关键字 1) 关于final的重要知识点 final关键字可以用于成员变量.本地变量.方法以及类. final成员变量必须在声明的时候初始化或者在构造器中初始化,否则就会报编译错误. ...

  10. bookStore案例第一篇【部署开发环境、解决分类模块】

    前言 巩固Servlet+JSP开发模式,做一个比较完整的小项目 成果图 该项目包含了两个部分,前台和后台. 前台用于显示 后台用于管理 该项目可分为5个模块来组成:分类模块,用户模块,图书模块,购买 ...