在写驱动程序和应用程序时,经常看到,有些时候用fread,read, fwite,wirte,是啥原因呢?
linux系统对文件的操作有带缓存的IO操作和不带缓存的IO操作。
带缓存的函数:
fwrite,fopen,fclose,fwrite,fread
这里的带缓存指在用户层带缓存,在内核也是带缓存的,内核是有缓存器的,只是用户看不到而已。带缓存IO也叫标准IO,符合ANSI C 的标准IO处理,不依赖系统内核,所以移植性强,带缓存IO其实就是在用户层再建立一个缓存区,这个缓存区的分配和优化长度等细节都是标准IO库代你处理好了,不用去操心。
不带缓存的函数:
wrtie,open,close,write,read
这里的不带缓存指在用户层不带缓存,在内核也是带缓存的。
以文件I/O函数write为例,size_t write (int fd,const void * buf,size_t count);用户向文件写入数据时,首先要将数据写入内核的缓冲存储器,等存储器满才写入文件中,假设内核缓存区长度为100个字节,用户需要写10次才能够写满,写满后,内核才将数据真正写入文件。
那区别是啥呢。
由上面的不带缓存函数write例子知道需要十次才能够写入文件,如果是带缓存的函数呢,假设用户层缓存区(流缓存区)为50个字节,调用c标准函数fwrite先将数据写入流缓存区,若流缓存区满,再将数据写入内核缓存区,写2次到内核的缓冲存储器,这样可以减少对内核缓冲寄存器的次数,这样对于大量数据写入文件时,fwrite函数的优势显现出来了,能够提供效率。
带缓存函数和不带缓存函数写入文件的操作流程如下:
无缓存IO操作数据流向路径:数据——内核缓存区——磁盘
标准IO操作数据流向路径:数据——流缓存区——内核缓存区——磁盘