最近有个需求,一个老的app,在android11的机子上运行各种闪退,要求对这个老项目适配android11系统。
项目拿到手发现闪退的节本原因都是各种文件创建不存在,网上查找了一下原因,基本确定就是文件分区存贮这个新特性导致的。
这个文件分区存贮其实在android10就已经存在,只不过到了android11的机子上,这个特性就强制执行了。
具体到项目代码上就是类似下面代码:
Stirng filePath = Environment.getExternalStorageDirectory();
File destFile = new File(filePath ,"test.txt");
FileOutputStream fileOut = new FileOutputStream(destFile);
执行到FileOutputStream的时候就会报错,表示destFile其实不存在,最坑的是上面的两行代码都不会报错,只有开始创建输出流的时候才会报错:文件不存在。
解决办法其实很简单,就是改成如下代码就可以正常运行:
Stirng filePath = context.getExternalFilesDir("").getAbsolutePath();
File destFile = new File(filePath ,"test.txt");
FileOutputStream fileOut = new FileOutputStream(destFile);
要理解这个现象,就要对分区存储这个特性有个直观的认识:
说白了就是context.getExternalFilesDir("").getAbsolutePath()这个路径下创建文件等操作就没有问题,因为这个目录获取的是
/storage/emulated/0/Android/data/包名/files
这个目录是app自己“私有”的目录,所以只要在这个目录下,android11以前版本的所有的其他操作都没有问题。
,
而Environment.getExternalStorageDirectory()这个方法获取的路径是
/storage/emulated/0/
这个目录下的目录都属于“公有”目录范围,如果我们的app需要读写这个目录下的文件,在android11系统当中就需要新的权限和新的访问方式。
所以,涉及到app自己操作自己所需要的文件的时候,只需要梳理一下以往在公有目录下操作文件的代码,然后全部替换成在app自己的私有目录下操作文件即可。
至于涉及公有目录的操作,比如拍照等等,网上文章很多,这里就不做赘述了。