AndroidQ文件存储适配

家住魔仙堡 提交于 2019-12-28 01:34:23

安卓Q即安卓10.0已经发布多时,不过大多数开发者并没有真机去测试,最近各厂商系统陆续推送了10.0的升级,因此必须要考虑去适配10.0系统了(建议大家先查看安卓Q系统权限变更相关文章,这里只说存储权限的适配方法,不做详细介绍)!

关于10.0系统权限方面的改变,大家可以搜索相关文章,这里主要讲一下存储权限的变化,10.0之前我们在保存或者查询文件时,首先需要申请存储权限:

    <uses-permission android:name="com.android.launcher.permission.READ_SETTINGS" />
    <uses-permission android:name="com.android.launcher.permission.WRITE_SETTINGS" />

但是,在10.0(targetSdkVersion=29)系统中,该权限已经不再起作用。应用中的行为表现为,即便你开启了存储权限,当你检测是否开启时,返回的结果是未开启,所以当你在把targetSdkVersion设为29或更高时,就一定要考虑这个问题了!对于暂时不想适配的,又不影响应用运行的方法,其它文章也有介绍,比如:targetSdkVersion设置为29以下,以及:

<application android:allowExternalStorageSandbox="false" ... >
  </application>

等,但这些方法都是暂时的,过后的版本,不论你怎么设置,都无法再使用10.0以前的文件存储方式了,也就是说你必须要适配安卓Q即10.0!

Q的存储方式变化,即引入了沙盒机制,应用可以随意访问自身在沙盒内创建的文件夹及文件,不需要任何权限,且在沙盒内创建的文件夹及文件会随着应用的卸载一并删除。沙盒路径为:

内部存储/Android/data/com.xx.xx(应用包名)/files/

当你为应用创建沙盒文件时,可去这里查看!

需要注意的是,沙盒里的文件并不能对外显示,比如Q以前,我们保存图片后,去相册里查看,立马就能看到刚刚保存的图片,但在Q系统中,保存图片到沙盒后,再去相册中查看是看不到的,这里需要一个骚操作,是保存到沙盒中后,需要再手动复制一份到公共文件夹中(公共文件夹包括:Downloads、Documents、Pictures 、DCIM、Movies、Music、Ringtones 等),在公共文件夹中创建的文件在应用卸载时是不会被删除的。

因此,我们在保存图片,音视频,其它文件时,需要对外显示的,就可以复制一份到相应的公共文件夹,不需要对外显示的就不用动,应用内部显示自己保存的内容时,就直接访问自己的沙盒中的目录就可以了!

具体方法:
保存图片:

String filepath=context.getExternalFilesDir(Environment.DIRECTORY_PICTURES) + "/" + 自己定义的文件夹名称+ "/";
String filename="xxx.png";
File imageFile = new File(filepath, filename);//这一步,系统会自动为你在沙盒中创建文件夹
...此处为保存(下载)图片的方法...

下载完成后,可以去上述文件夹路径查看是否保存成功!若要对外显示,则需要复制到公共文件夹:

    @RequiresApi(api = Build.VERSION_CODES.Q)
    public static void copyPrivateImgToCommen(Context context, String orgFilePath, String displayName) {
        ContentValues values = new ContentValues();
        values.put(MediaStore.Files.FileColumns.DISPLAY_NAME, displayName);
        values.put(MediaStore.Files.FileColumns.TITLE, displayName);
        values.put(MediaStore.Files.FileColumns.MIME_TYPE, "image/*");
        values.put(MediaStore.Images.Media.RELATIVE_PATH, Environment.DIRECTORY_PICTURES + "/" + 自己定义的文件夹名称);
        Uri external = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
        ContentResolver resolver = context.getContentResolver();
        Uri insertUri = resolver.insert(external, values);
        InputStream ist = null;
        OutputStream ost = null;
        try {
            ist = new FileInputStream(new File(orgFilePath));
            if (insertUri != null) {
                ost = resolver.openOutputStream(insertUri);
            }
            if (ost != null) {
                byte[] buffer = new byte[4096];
                int byteCount = 0;
                while ((byteCount = ist.read(buffer)) != -1) {
                    ost.write(buffer, 0, byteCount);
                }
            }
        } catch (IOException e) {

        } finally {
            try {
                if (ist != null) {
                    ist.close();
                }
                if (ost != null) {
                    ost.close();
                }
            } catch (IOException e) {

            }
        }
    }

此方法参考自其它开发者文章,主要在于这几个字段:
MediaStore.Files.FileColumns.MIME_TYPE:文件类型,图片即“image/”,视频即“video/”,
MediaStore.Images.Media.RELATIVE_PATH:存储路径,图片即MediaStore.Images,视频即MediaStore.Video。
注意:即便是在自己的沙盒中,保存图片和视频等,也要根据文件类型,选择对应的文件夹如:Environment.DIRECTORY_PICTURES或者Environment.DIRECTORY_MOVIES。

保存视频:方法同保存图片,但要注意区分文件类型及存储路径!

如何查询自己保存的文件?以图片为例:

            File privateFile = getExternalFilesDir(Environment.DIRECTORY_PICTURES + "/自己定义的文件夹名称");
            File[] files = privateFile.listFiles();
            if (files != null) {
                for (File file : files) {
                     //file 即你保存的图片文件
                }
            }

其它类型,就取对应的Environment.DIRECTORY_路径!

写在最后,以上方法都是在Q系统下的操作方法,适配时需要判断系统版本号,Build.VERSION.SDK_INT<Q时,依然要申请存储权限,并按Q之前的操作方法去操作,>=Q时,则无需再申请权限,按Q操作方式操作即可!

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!