美文网首页Android
android存储分区和android11填坑

android存储分区和android11填坑

作者: only_run | 来源:发表于2021-07-03 11:36 被阅读0次

    android的物理存储划分

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

    • 内部存储
      设备中每一个安装的 App,系统都会在内部存储空间的 data/data 目录下以应用包名为名字自动创建与之对应的文件夹,这个文件夹也用来 存放SharedPreferences 和 SQLiteDatabase 的数据, App 中的 WebView 缓存页面信息也在这文件夹下;
      但是 当app被卸载的时候,这个文件夹 会被删除掉。
      开发过程中,可通过 Context对象提供的 API 读取操作 内部存储中的文件
    //内部存储
                /**
                 * 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)
                }
    
    • 外部存储
      一般的手机设备都有会内置 SD 卡,同时也提供 SD 卡的拓展,可能对应路径的目录名有所差异。
      所说的外部存储,就是手机设备内置的SD卡和扩展的SD卡 提供的存储空间;
      外部存储 也会为 安装的app 提供一块区域(文件夹) ,用来存放私有的文件
      通常的路径是:/storage/emulated/0/Android/data/包名/

    什么是存储分区

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

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

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

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

    • app私有区域
      文件目录在/android/data/app包名 下,APP自身 不需要读写权限 就可以进行读写,其他的app是无法访问这一区域的(自android7.0起);开启存储分区后 无变化
    • app共享区域
      只要有读写权限 每个app都可以读写,文件目录有DCIM、Pictures,Download等在/storage/emulated/0 下的文件夹及文件;开启存储分区后,无法直接通过File 访问共享区域的文件,需要使用FileProvider或者MediaStore 来进行访问。

    例如:在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" 就是 授权认证的信息,如果填写的和清单文件中的不一致 会 导致文件读写时 错误;

    相关文章

      网友评论

        本文标题:android存储分区和android11填坑

        本文链接:https://www.haomeiwen.com/subject/ejxyultx.html