Android 10文件存储适配

Android 10文件存储适配

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

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

  <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <uses-permission
        android:name="android.permission.WRITE_EXTERNAL_STORAGE"
        tools:ignore="ScopedStorage" />

但是,在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);//这一步,系统会自动为你在沙盒中创建文件夹
...此处为保存(下载)图片的方法...

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操作方式操作即可!

 

补充:
安卓Q如何进行选择相册和拍照呢?又如何进行文件上传操作呢?

如果只用来显示,那么只需要获取到图片/视频Uri就可以了,但是如果要获取到file进行上传操作呢?如果你还是用Q以前的方法:File file=new File(path);的话你会发现,根本无法获取到这个文件,我们只能通过曲线救国的方法,即拍照和选取相册完成后,得到图片uri,并通过该uri转成file的形式,将file保存在沙盒文件夹内,然后再去沙盒文件夹内取file,再上传!也就是需要把你要上传的文件复制一份到沙盒…呵呵!

另外,拍照时调用相机方法有变化,需要判断系统版本号,Q系统需要通过Uri形式获取到拍照结果:

            ContentValues values = new ContentValues();
            values.put(MediaStore.Images.Media.DISPLAY_NAME, filename);
            values.put(MediaStore.Images.Media.MIME_TYPE, "image/*");
            values.put(MediaStore.Images.Media.RELATIVE_PATH, Environment.DIRECTORY_PICTURES);
            uri = context.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);

需要记录这个Uri,拍照结果在onActivityResult中获取:
Q系统通过url转File的形式得到图片路径:

    public static String uriToFile(Context context, Uri uri, String privateFileName) {
        File privateFile = new File(getImagePathCache(context), privateFileName + ".png");
        File fileParent = privateFile.getParentFile();
        if (!fileParent.exists()) {
            fileParent.mkdirs();
        }
        if (privateFile.exists()) return privateFile.getAbsolutePath();
        InputStream ist = null;
        OutputStream ost = null;
        try {
            privateFile.createNewFile();
            ist = context.getContentResolver().openInputStream(uri);
            ost = new FileOutputStream(privateFile);
            byte[] buffer = new byte[4096];
            int byteCount = 0;
            while ((byteCount = ist.read(buffer)) != -1) {  // 循环从输入流读取 buffer字节
                ost.write(buffer, 0, byteCount);        // 将读取的输入流写入到输出流
            }
        } catch (IOException e) {
            return "";
        } finally {
            try {
                if (ist != null) {
                    ist.close();
                }
                if (ost != null) {
                    ost.close();
                }
            } catch (IOException e) {
                return "";
            }
        }
        return privateFile.getAbsolutePath();
    }

其中的getImagePathCache方法就是你的沙盒文件夹路径:

context.getExternalFilesDir(Environment.DIRECTORY_PICTURES) + "/cache/";//cache是自定义的文件夹名称

相册选择结果同样在onActivityResult中获取:
相册选择的结果可以直接通过onActivityResult方法返回的intent.getData()获取,然后调用上述uri转File方法,得到图片路径!

如果考虑这种缓存文件占用内存空间,则在使用后删除即可!


原文链接:https://blog.csdn.net/baiyuliang2013/article/details/103735264

上一篇:Android R 存储机制变更


下一篇:一种非常简单的Android屏幕适配方案,讲的明明白白!