Android

android存储分区和android11填坑

2021-07-03  本文已影响0人  only_run

android的物理存储划分

安卓设备的物理存储 分为两大块,内部存储和外部存储

//内部存储
            /**
             * Environment.getDataDirectory() : /data
             * context.getFilesDir() : /data/user/0/com.e.dk_wd/files
             * context.getCacheDir() : /data/user/0/com.e.dk_wd/cache
             * context.getDataDir() : /data/user/0/com.e.dk_wd
             */
            Log.i("----", "Environment.getDataDirectory() : " + Environment.getDataDirectory().absolutePath)
            Log.i("----", "context.getFilesDir() : " + this.filesDir!!.absolutePath)
            Log.i("----", "context.getCacheDir() : " + this.cacheDir!!.absolutePath)
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                Log.i("----", "context.getDataDir() : " + this.dataDir!!.absolutePath)
            }

什么是存储分区

存储分区,是android系统对APP访问外部存储 添加了限制;开启存储分区后 APP只能访问自己目录下的文件和公共文件,
需要特别指出的是android 10 虽支持存储分区 但可不开启,对于android11 来说,必须开启存储分区,android11必须使用存储分区,
即使设置 android:requestLegacyExternalStorage="true" 也无效

//android10(包括android10)以下 可在清单文件中声明 android:requestLegacyExternalStorage="true" 来不启用存储分区
<application
    ...
        android:requestLegacyExternalStorage="true">

开启存储分区后对文件访问的影响

对原来的内部存储没有什么影响,但是对外部存储有影响;
外部 存储有两个区域 app私有区域和app共享区域;

例如:在android 10以下的手机设备上 调起相机拍照,使用 Uri.fromFile 的方式来创建照片文件是 没有问题的,但是 android11上,即使调起了相机拍照后 也无法成功保存照片

   if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
      //android7开始 就支持FileProvider,所以这里判断大于android7
       mCurrentPhotoUri = FileProvider.getUriForFile(mContext,
                            "com.xx.fileprovider", photoFile);
 } else {
         mCurrentPhotoUri = Uri.fromFile(photoFile);
  }

什么是FileProvider?

android 7 后支持 应用之间共享文件,这就是FileProvider的作用;
使用FileProvider 需要在清单文件中 声明,注意一下 android:authorities指定的属性值,尽可能的保证唯一性 一般以".fileProvider" 结尾

<provider
            android:name="androidx.core.content.FileProvider"
            android:authorities="${applicationId}.fileProvider"
            android:exported="false"
            android:grantUriPermissions="true"
            tools:replace="android:authorities">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                <!--这个file_paths文件在 Resource/xml 下-->
                android:resource="@xml/file_paths"
                tools:replace="android:resource" />
</provider>
       

其中 file_paths文件在 Resource/xml 下,原来指定共享的 文件路径;

<?xml version="1.0" encoding="utf-8"?>
<paths>
    <files-path
        name="file_path"
        path="." />
    <cache-path
        name="cache_path"
        path="." />
    <external-path
        name="external_path"
        path="." />
    <external-files-path
        name="external_files_path"
        path="." />
    <external-cache-path
        name="external_cache_path"
        path="." />
    <!--
    每个节点都支持两个属性:name+path;
    //path:需要临时授权访问的路径(.代表所有路径)
    //name:就是你给这个访问路径起个名字
    例如:
    <root-path name="root" path="" /> //代表设备的根目录new File("/");
    <files-path name="files" path="" /> //context.getFilesDir()
    <cache-path name="cache" path="" /> //context.getCacheDir()
    <external-path name="external" path="" /> //Environment.getExternalStorageDirectory()
    <external-files-path name="name" path="path" /> //context.getExternalFilesDirs()
    <external-cache-path name="name" path="path" /> //getExternalCacheDirs()
    -->
</paths>

然后使用FileProvider API 来生成File的Uri路径;

FileProvider.getUriForFile(mContext,"com.xx.fileprovider", photoFile);

这个"com.xx.fileprovider" 就是 授权认证的信息,如果填写的和清单文件中的不一致 会 导致文件读写时 错误;

上一篇下一篇

猜你喜欢

热点阅读