引言
我们之所以倾向于使用标准IO不仅是其功能比系统IO丰富,还有是因为标准IO有个很显著的地方就是标准IO拥有缓冲功能,能够提高程序的吞吐量。那么这篇笔记主要是研究标准IO是什么时候刷新数据到内核:
标准IO缓冲区的三种类型
一、不缓冲
虽然说标准IO分配了缓冲区buffer,但是这种方法并不缓冲就直接将数据刷新了。那么一般会用到这种类型的数据:紧急数据(出错处理)
举个例子:在打开屏幕这个文件的时候,有两个描述符指向屏幕,分为:stdout、stderr
stderr为错误信息,那么这种情况并不应该缓冲,而是应该越快越好去让内核处理。
下面我们通过代码来验证一下:
程序运行结果:
按照先后顺序应该是先输出小写的abcd,而结果则是输出大写的ABCD
二、全缓冲
这种类型的意思是:要将分配的缓冲区buffer的大小全部填满,才进行数据的刷新。
那么遇到这种设定可能会照成很大的不便,因此就产生了很多让全缓冲模式下也能强制让数据刷新的功能,那么下面一次性总结一下让全缓冲模式下刷新数据的条件:
- 数据填满缓冲区
- fflush()函数
- fclose()函数
- 正常退出程序
- setbuf() / setvbuf() 这个函数主要是设置缓冲区的大小
下面我们用代码来验证一下:
首先我们来看一下无缓冲中的例子:
小写的abcd一样从缓冲区刷新了,原因就在于程序正常退出了,自动刷新缓冲区。
下面是验证数据填满缓冲区:
程序运行结果:
如果将字节改为2500:
程序运行结果:
写入了2048个字节到a.txt
下面来验证一下使用setbuf函数与setvbuf函数:
那么这次就成功将200字节写入到文件中
下面使用Setvbuf来做实验:首先要看一下setvbuf函数怎么用
需要注意的是,使用这个函数需要自己预先定义一个缓冲区Buf,并且设置其大小。这个缓冲区与系统自己默认分配的缓冲区并不相同。
名称mode | 功能 |
---|---|
_IONBF | 无缓冲 |
_IOLBF | 行缓冲 |
_IOFBF | 全缓冲 |
程序代码:
程序结果:
只写入了250个字节
三、行缓冲
在缓冲区一行数据是有特定的大小的,行缓冲就是数据在缓冲区收集够一行了,就开始对数据进行刷新。那么让行缓冲刷新数据的条件:
- 数据填满缓冲区
- fflush()函数
- fclose()函数
- 正常退出程序
- setbuf() / setvbuf() 这个函数主要是设置缓冲区的大小
- 遇到换行符’\n’
下面我们用代码来验证一下:(修改无缓冲的例子)
在无缓冲的例子中,加上换行字符
运行结果:
因为stdout遇到了换行符,所以直接使用行缓冲的方式将数据刷新了