23. fread和fwrite的读写缓冲区
fwrite(buff, sizeof(buff), 1, pFile);//返回实际写入的完整项数,参数3尽量填1,简单粗暴 fread(buff, sizeof(buff), 1, pFile);//返回实际读取的完整项数,参数3尽量填1,简单粗暴
文件内容:
每4字节复制旧文件到新文件,忽略错误检查:
FILE *pFileSource = fopen("C++ Primer.exe", "rb");//CreateFile FILE *pFileDest = fopen("C++ Primer2.exe", "wb"); char buffSource[16] = { 0 };//设置很小的缓冲区,便于测试 char buffDest[16] = { 0 }; setvbuf(pFileSource, buffSource, _IOFBF, 7);//只给7字节,便于测试 setvbuf(pFileDest, buffDest, _IOFBF, 7); fseek(pFileSource, 0, SEEK_END); long fileSize = ftell(pFileSource); rewind(pFileSource); unsigned int buff = 0; for (long i = 0; i < fileSize / 4; i++) { fread(&buff, sizeof(buff), 1, pFileSource);//ReadFile fwrite(&buff, sizeof(buff), 1, pFileDest);//直接写入缓冲区,满了调用flush } unsigned char byte = '\0'; for (long i = 0; i < fileSize % 4; i++) { fread(&byte, sizeof(byte), 1, pFileSource); fwrite(&byte, sizeof(byte), 1, pFileDest); } fflush(pFileDest);//WriteFile fclose(pFileSource);//CloseHandle fclose(pFileDest);
以上代码将缓冲区大小设为7,但是实际向下舍入到 2的整数倍,即6字节。
第1次fread读取6字节填满缓冲区,再取出4字节给代码中的unsigned int buff = 0;此变量在0019FF0C处。
F10单步,第1次fwrite写入4字节到写缓冲区,watch切换到pFileDest观察。再观察要写入的文件,发现大小没写入任何数据,因为写缓冲区没满。
F10单步,第2次fread读取6字节
F10单步,第2次fwrite,0506写入缓冲区末尾,这时满了,将整个6字节缓冲区写入文件。再调整_ptr到_base,并把0708写入缓冲区。打开文件,发现有6字节数据。
F10单步,第3次fread,缓冲区数据足够,未读入新数据,有变化的是_ptr
F10单步,第3次fwrite,刚好填满写缓冲区,打开文件发现依旧是6字节,未写入。
F10单步,第4次fread,读入6字节新数据
F10单步,第4次fwrite,缓冲区已满,调用fflush,打开文件发现多了6字节
总结:第一次fread和fwrite时会分别提供默认4096字节的读和写缓冲区,也可以自己调用setvbuf函数设置缓冲区。fread从文件读取_bufsiz大小的数据存放在读缓冲区,后续如无必要,不会读文件,而是直接读缓冲区。同理,fwrite将数据写入写缓冲区,除非缓冲区已满,否则不写入文件。两者都通过移动指针_ptr读取缓冲区和写入缓冲区。