Android FileProvider详细解析和10.0的适配

今天项目中文件存储的时候,我们都知道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文件


<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)
Android FileProvider详细解析和10.0的适配_第1张图片

Android10.0 临时解决方案
如果适配兼容10.0的文件存储比较麻烦,可以采用临时方案:

Android FileProvider详细解析和10.0的适配_第2张图片

 android:requestLegacyExternalStorage="true"

你可能感兴趣的:(Android)