本文是基于hadoop-2.8.0的源码实现
1. linux上的sync, fsync, fdatasync的区别
sync会将输出流缓冲区数据排入到OS维护的写磁盘对列,而不会等在磁盘操作完成,因此sync调用返回后数据依然有可能会丢失。
fsync和sync不同的是它会等待写磁盘操作完成,同时它还会修改文件meta信息。因此fsync往往对应两次磁盘IO:一次写数据到磁盘,一次修改文件meta信息到磁盘。
fdatasync 和fsync一样等到磁盘操作完成,但是它不会修改文件的meta信息。
2 FsDataOutputStream hflush和hsync
回到hdfs的hflush和hsync, 文件的meta信息存放在namenode上,文件的data信息存放在datanode上。
client调用hflush和hsync流程都是一样的,首先这里再次说明一下write的流程:
client端:
输出流底层会缓冲数据,缓冲到一定大小后flush到datanode,这个缓冲区是一个block。
写到block的数据又被切分成一个个packet,发送block的时候是一个个packet发送,发送完packet后会将packer移到等待确认对列中,然后等待datanode的确认。
datanode端:
写数据采用pipeline机制,比如数据需要在3个datanode(dn1, dn2, dn3)上备份,那么dn1收到 client端的packet后,写packet到dn2, 然后将packet写到本地磁盘并flush, dn2如法炮制写dn3并写本地。
确认