《Unix环境高级编程》中提到函数sync,fsync,fdatasync对缓存向磁盘写入数据做了解释:
当将数据写入文件时,内核通常先将该数据复制到其中一个缓冲区中,如果该缓冲区尚未写满,则并不将其排入输出队列,而是等待其写满或者当内核需要重用该缓冲区以便存放其他磁盘块数据时,再将该缓冲排入输出队列,然后待其到达队首时,才进行实际的I/O操作。这种输出方式被称为延迟写。
因此要实现缓存写入磁盘完成,必须对缓存和磁盘进行同步:
使用函数有:
#include <unistd.h>
void sync(void);
int fsync(int fd);
int fdatasync(int fd);
以上函数的fd参数需要通过linux系统函数open,create来打开或获得。
void sync(void),只是将修改的块缓存排入写队列,并不等磁盘写结束,对应于linux程序sync直接运行,即命令行 sync
int fsync(int fd),对fd指定的一个文件起作用,并且等写磁盘操作结束才返回,更新数据和文件属性。对应于linux程序sync直接运行,即命令行 sync filepath
int fdatasync(int fd),类似于fsync,只更新数据。对应于linux程序sync直接运行,即命令行 sync filepath, sync内部会对数据进行判断。
拷贝普通文件的函数:
#include "fcntl.h"
#include "unistd.h"
bool copyFile(const char *srcPath, const char *dstPath)
{
int srcFd = open(srcPath, O_RDONLY);
int dstFd = open(dstPath, O_WRONLY | O_CREAT);
if (srcFd == -1 ) return false;
if (dstFd == -1) return false;
char buf[4096]; //一般页的大小
int n;
while ((n = read(srcFd, buf, 4096)) > 0)
{
write(dstFd, buf, n);
}
fsync(dstFd); //实现文件写入同步
close(srcFd);
close(dstFd);
return true;
}
附上sync源代码的地址:
https://github.com/coreutils/coreutils/blob/2ed7c2867974ccf7abc61c34ad7bf9565489c18e/src/sync.c