赞
踩
今天项目中文件存储的时候,我们都知道7.0以上要通过FileProvider保存文件,但是在小米10pro上报错,无法找到文件路径,对各个系统的反复测试,只有Android Q的手机会出现异常,于是百度才发现是Android10.0 更改了文件存取机制。
Android7.0 (N) 开始,将严格执行 StrictMode 模式,也就是说,将对安全做更严格的校验。而从 Android N 开始,将不允许在 App 间,使用 file:// 的方式,传递一个 File ,否者会抛出 FileUriExposedException的错误,会直接引发 Crash。
但是,既然官方对文件的分享做了一个这么强硬的修改(直接抛出异常),实际上也提供了解决方案,那就是 FileProvider,通过 content://的模式替换掉 file://,同时,需要开发者主动升级 targetSdkVersion 到 24 才会执行此策略。
FileProvider是android support v4包提供的,是ContentProvider的子类,便于将自己app的数据提供给其他app访问。
在app开发过程中需要用到FileProvider的主要有:
1、配置AndroidManifest文件
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="包名.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
authorities:一个标识,在当前系统内必须是唯一值,一般用包名。
exported:表示该 FileProvider 是否需要公开出去。
granUriPermissions:是否允许授权文件的临时访问权限。这里需要,所以是 true。
2、在res的建xml目录,放入file_paths.xml文件
<?xml version="1.0" encoding="utf-8"?>
<resources>
<paths>
<external-path name="image" path="."/>
</paths>
</resources>
这里主要对几个路径做个概括:
root-path对应device_root,也就是File file = new File("/"),即根目录,一般不需要配置。
files-path对应 content.getFileDir() 获取到的目录。
cache-path对应 content.getCacheDir() 获取到的目录
external-path对应 Environment.getExternalStorageDirectory() 指向的目录。
external-files-path对应 ContextCompat.getExternalFilesDirs() 获取到的目录。
external-cache-path对应 ContextCompat.getExternalCacheDirs() 获取到的目录。
对应关系为:
TAG | Value | Path |
---|---|---|
TAG_ROOT_PATH | root-path | / |
TAG_FILES_PATH | files-path | /data/data/<包名>/files |
TAG_CACHE_PATH | cache-path | /data/data/<包名>/cache |
TAG_EXTERNAL | external-path | /storage/emulate/0 |
TAG_EXTERNAL_FILES | external-files-path | /storage/emulate/0/Android/data/<包名>/files |
TAG_EXTERNAL_CACHE | external-cache-path | /storage/emulate/0/Android/data/<包名>/cache |
3.Java代码获取URI
File f = new File(path,"img-"+curDateTimeStr +".png"); Uri uri; if(Build.VERSION.SDK_INT>= 24) { //判读版本是否在7.0以上 //参数1 上下文, 参数2 Provider主机地址 和配置文件中保持一致 参数3 共享的文件 uri =FileProvider.getUriForFile(this, "com.noway.android.foreverlove.fileprovider", f); //添加这一句表示对目标应用临时授权该Uri所代表的文件 }else{ uri = Uri.fromFile(f); } try { OutputStream out = getContentResolver().openOutputStream(uri); bm.compress(Bitmap.CompressFormat.PNG, 2, out); out.flush(); out.close(); Intent service = new Intent(this, ImageUploadService.class); service.putExtra("path",f.getPath()); startService(service); } catch (FileNotFoundException e) { e.printStackTrace(); Toast.makeText(this,e.getMessage(),Toast.LENGTH_SHORT).show(); } catch (IOException e) { e.printStackTrace(); }finally { }
Android 10.0的手机会抛出异常:java.io.FileNotFoundException: open failed: ENOENT (No such file or directory)
Android10.0 临时解决方案
如果适配兼容10.0的文件存储比较麻烦,可以采用临时方案:
android:requestLegacyExternalStorage="true"
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。