Android FileProvider详细解析和10.0的适配

本文探讨了在Android 10.0(Q)上遇到的文件存储问题,特别是在小米10 Pro设备上出现的文件路径找不到的异常。深入分析了Android 10.0更改的文件存取机制,并提供了使用FileProvider的详细步骤来替代file://方式,确保应用程序在高版本Android系统上的正常运行。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

今天项目中文件存储的时候,我们都知道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() 获取到的目录。

对应关系为:

TAGValuePath
TAG_ROOT_PATHroot-path/
TAG_FILES_PATHfiles-path/data/data/<包名>/files
TAG_CACHE_PATHcache-path/data/data/<包名>/cache
TAG_EXTERNALexternal-path/storage/emulate/0
TAG_EXTERNAL_FILESexternal-files-path/storage/emulate/0/Android/data/<包名>/files
TAG_EXTERNAL_CACHEexternal-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"
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值