android 设备写入文件,立即断电重启后,文件丢失,数据没有保存问题

在android开发的过程中碰到写入文件后,立即断电重启,发现写入的文件丢失了

写入时检查了,写入是没有失败的,经过查找资料可能是如下问题引起:

  Linux/Unix系统中,在文件或数据处理过程中一般先放到内存缓冲区中,等到适当的时候再写入磁盘,以提高系统的运行效率。


可能是因为断电时,文件没有写入的物理介质中导致,解决办法如下:

       在write/fwrite写入后,添加fsync(), 这样可以将缓存中的内容强制写入到磁盘中


关于write/fwrite和fsync的关系如下:

read/write/fsync: 

1. Linux底层操作;

2. 内核调用, 涉及到进程上下文的切换,即用户态到核心态的转换,这是个比较消耗性能的操作。

fread/fwrite/fflush: 

1. C语言标准规定的io流操作,建立在read/write/fsync之上

2. 在用户层, 又增加了一层缓冲机制,用于减少内核调用次数,但是增加了一次内存拷贝。


两者之间的关系,见下图:

android 设备写入文件,立即断电重启后,文件丢失,数据没有保存问题_第1张图片


补充: 

1. 对于输入设备,调用fsync/fflush将清空相应的缓冲区,其内数据将被丢弃;

2. 对于输出设备或磁盘文件,fflush只能保证数据到达内核缓冲区,并不能保证数据到达物理设备, 因此应该在调用fflush后,调用fsync(fileno(stream)),确保数据存入磁盘。

3. 在android java层中,需要调用FileDescriptor的sync()方法来确保数据存入磁盘。

/* Java 示例代码 */ 
FileOutputStream fos = null;
byte[] buf = {1,2,3,4};
try {
    fos = new FileOutputStream("/sdcard/test.txt");

    //将buf中的数据写入fos
    fos.write(buf);

    //将fos的数据保存到内核缓冲区
    //不能确保数据保存到物理存储设备上,如突然断点可能导致文件未保存
    fos.flush();

    //将数据同步到达物理存储设备
    FileDescriptor fd = fos.getFD();
    fd.sync();

} catch(Exception e) {
    e.printStackTrace();
} finally {
    if(fos!=null)
        fos.close();
}

你可能感兴趣的:(android)