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. 在用户层, 又增加了一层缓冲机制,用于减少内核调用次数,但是增加了一次内存拷贝。

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

补充:

  1. 对于输入设备,调用 fsync/fflush 将清空相应的缓冲区,其内数据将被丢弃;
  2. 对于输出设备或磁盘文件,fflush 只能保证数据到达内核缓冲区,并不能保证数据到达物理设备, 因此应该在调用 fflush 后,调用 fsync(fileno(stream)),确保数据存入磁盘。
  3. 在 android java 层中,需要调用 FileDescriptor 的 sync()方法来确保数据存入磁盘。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
/* 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 设备写入文件,立即断电重启后,文件丢失,数据没有保存

http://www.kch8.top/2022/11/02/android-write-file-fail/

发布于

2022-11-02

更新于

2023-08-18

许可协议

评论