Android 10 开始引入了沙盒机制,应用在 sdcard 中默认只能读写私有目录(即/sdcard/Android/data/[应用包名]/)。除非在 AndroidManifest.xml 下声明以下属性:
<application
...
android:requestLegacyExternalStorage="true">
这样的话就会暂时停用沙盒机制,正常读写/sdcard下文件。
而从Android 11开始,/sdcard/Android/data 和 /sdcard/Android/obb 在获取了所有文件权限之后还是无法读写,需要通过SAF授权目录并且通过DocumentFlie来获取文件
在Android 12和13中授权之后获取了私有目录的文件的路径之后,可以直接通过new File(path)来获取到该文件,但是path 的格式必须是
String path = "/storage/emulated/0/Android/data/xxx/xxx.pdf";
path可以使用以下方法来进行转换
//假如这是通过DocumentFile dir遍历出来的文件
DocumentFile f;
if(f.isFile()&&f.exists()){
// f.getUri().getPath()的路径:
// /tree/primary:Android/data/xxx/document/primary:Android/data/xxx/xxx.pdf
String path = f.getUri().getPath();
//使用正则表达式替换路径
// /storage/emulated/0/Android/data/xxx/xxx.pdf
path = path.replaceAll(".*?/document/primary:","/storage/emulated/0/");
}
//在Android 12和13中授权之后获取了私有目录的文件的路径之后,可以直接通过new File(path)来获取到该文件
File file1 = new File(path);
//如果文件存在的话为true
System.out.println(file1.exists());
并且在Android 11中授权目录获取到了文件的路径之后,无法通过new File(path)来获取到该文件,还得通过DocumentFile.fromSingleUri(context,uri)来获取一个DocumentFile,读取文件。
但是如果必须要用File来操作文件的话(比如有时候一些工具类是使用File来进行操作的,DocumentFile不是继承File的,所以不能直接拿来用),因为无法通过new File(path)来获取到该文件,我们可以通过使用DocumentFile拷贝一份文件到自己的私有目录下,这样以后读取的就是自己私有目录下的文件,不会有权限问题,就可以通过new File(path)来访问该文件进行操作(我愿称之为曲线救国的操作)
//延续上面的代码
File file2 = new File(path);
//Android 11中直接通过new File()获取的Android/data下的文件路径是没有权限的
//这个判断始终为true
if (!file2.exists()){
ToastUtils.showLongToast("文件不存在");
}
//可以通过拷贝一份到自己软件的私有目录中去
path = copyFileToPrivateStorage(context, f);
File file3 = new File(path);
//这样file3就在自己的私有目录下,是有权限访问的,就可以操作该文件了
//这是拷贝文件的方法
private String copyFileToPrivateStorage(Context context, DocumentFile sourceFile) {
InputStream in = null;
OutputStream out = null;
// 获取应用私有目录
File privateDir = context.getFilesDir();
File destFile = new File(privateDir, sourceFile.getName());
try {
// 打开输入流
in = context.getContentResolver().openInputStream(sourceFile.getUri());
// 打开输出流
out = new FileOutputStream(destFile);
// 复制文件
byte[] buffer = new byte[1024];
int length;
while ((length = in.read(buffer)) > 0) {
out.write(buffer, 0, length);
}
// 关闭流
in.close();
out.close();
} catch (IOException e) {
e.printStackTrace();
// 处理错误
}
// 文件复制成功,返回目标文件路径
return destFile.getPath();
}
参考文档:
感谢这位大佬分享的访问Android/data(obb)目录的方法