read/write/fsync与fread/fwrite/fflush的关系和区别
read/write/fsync:
1. linux底层操作;
2. 内核调用, 涉及到进程上下文的切换,即用户态到核心态的转换,这是个比较消耗性能的操作。
fread/fwrite/fflush:
1. c语言标准规定的io流操作,建立在read/write/fsync之上
2. 在用户层, 又增加了一层缓冲机制,用于减少内核调用次数,但是增加了一次内存拷贝。
两者之间的关系,见下图:
补充:
1. 对于输入设备,调用fsync/fflush将清空相应的缓冲区,其内数据将被丢弃;
2. 对于输出设备或磁盘文件,fflush只能保证数据到达内核缓冲区,并不能保证数据到达物理设备, 因此应该在调用fflush后,调用fsync(fileno(stream)),确保数据存入磁盘。
----------------------------------------------------------------------
fwrite,fflush,你不知道的事!
最近项目代码中遇到很多奇怪的问题,比如,在某个程序中用fwrite向配置文件中写入一些数据,而在另一段代码中需要读取该配置文件中的数据,写文件那段程序已经执行完了,但是数据并没有被写入文件,进而导致读文件的代码读不到数据,程序很难按理想的方式运行。这样下来将会导致一些列的问题,我今天就被这个问题纠结了一下午...
那么这个问题是什么原因导致的呢,该怎么解决呢?
导致问题的原因是 用fwrite函数写数据的时候,fwrite先将数据写到内存中的缓冲区内,等程序结束后才会将数据由缓冲区写入文件。所以我遇到的问题原因是整个函数还没有运行完毕,写入的数据还在缓冲区内。
解决办法就只有靠fflush()函数了。
#include<stdio.h>
int fflush(FILE *stream)
该函数的作用就是刷新缓冲区
做法是,在写入的数据在函数结束之前就需要的时候,调用fwrite等函数后,紧接着调用fflush()函数将缓冲区刷新,这样数据就会被立刻写入文件而不用等到程序结束(因为之前的数据都在缓冲区里)。
下面通过一个简单的例子让小伙伴们 加深理解:
- int main()
- {
- FILE *fp = fopen("./file.txt","a");
- if(fp == NULL)
- {
- printf("open file err!\n");
- return -1;
- }
- fwrite("hello",5,1,fp);
- //fflush(fp);
- close(fp);
- while(1)
- {
- sleep(5);
- }
- return 0;
- }
如上程序,当程序一直不退出的时候,虽然已经调用了fwrite但是数据却在缓冲区里,而并不在文件中,这时你可以查看改路径下的file.txt文件,你会发现文件已经建立但是大小为0,如果你在fwrite之后用fflush函数后数据就会立刻写入文件,我总结就这么多,不喜勿喷,希望对小伙伴们有帮助。
转自:http://blog.****.net/zhangxiong2532/article/details/50608898
http://blog.****.net/ybxuwei/article/details/22727565