Android 10文件存储适配

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

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

  

android:name="android.permission.WRITE_EXTERNAL_STORAGE"
         tools:ignore="ScopedStorage" />



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

 

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

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

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

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

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

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

具体方法:
(1)保存图片:
...此处为保存(下载)图片的方法(保存到App内部存储目录)...

//一定要写这个路径getExternalFilesDir(Environment.DIRECTORY_PICTURES),这是应用内部路径(可见的内部存储

String filepath=context.getExternalFilesDir(Environment.DIRECTORY_PICTURES) + File.separator + 自己定义的文件夹名称+ File.separator;
String fileName="xxx.png";

try {
   File dirs = new File(filepath);
   if (!dirs.exists()) {
      dirs.mkdirs();
   }
   fileUpdate = new File(dirs, fileName);
   if (!fileUpdate.exists()) {
      try {
         fileUpdate.createNewFile();
      } catch (IOException e) {
         e.printStackTrace();
      }
   }
   
   fileInputStream = new FileInputStream(file.getAbsolutePath());//读入原文件
   bufferedInputStream = new BufferedInputStream(fileInputStream);
   fileOutputStream = new FileOutputStream(fileUpdate.getAbsolutePath());
   bufferedOutputStream = new BufferedOutputStream(fileOutputStream);
   byte[] buffer = new byte[1024 * 2];
   int len;
   
   while ((len = bufferedInputStream.read(buffer)) != -1) {
      bufferedOutputStream.write(buffer, 0, len);
   }
   
   bufferedOutputStream.flush();
   bufferedInputStream.close();
   bufferedOutputStream.close();
} catch (IOException e) {
   e.printStackTrace();
}

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


@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 + File.separator + 自己定义的文件夹名称);
   Uri external = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
   ContentResolver resolver = context.getContentResolver();
   Uri insertUri = resolver.insert(external, values);
   InputStream is = null;
   
   BufferedInputStream bis = null;
   OutputStream os = null;
   BufferedOutputStream bos = null;
   try {
      is = new FileInputStream(new File(orgFilePath));
      bis = new BufferedInputStream(is);
      if (insertUri != null) {
         os = resolver.openOutputStream(insertUri);
         if (os != null) {
            bos = new BufferedOutputStream(os);
         }
      }
      if (os != null) {
         byte[] buffer = new byte[1024 * 2];//一次缓冲2k
         int len = 0;
         while ((len = bis.read(buffer)) != -1) {
            bos.write(buffer, 0, len);
         }
         bos.flush();
         bis.close();
         bos.close();
      }
   } catch (IOException e) {
      e.printStackTrace();
   }
}


此方法参考自其它开发者文章,主要在于这几个字段:
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.separator +自己定义的文件夹名称);
             File[] files = privateFile.listFiles();
             if (files != null) {
                 for (File file : files) {
                      //file 即你保存的图片文件
                 }
             }


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

写在最后,以上方法都是在Q系统下的操作方法,适配时需要判断系统版本号,Build.VERSION.SDK_INT=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.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 + File.separator +

自己定义的文件夹名称);


ContentResolver resolver = context.getContentResolver();
Uri insertUri = resolver.insert(external, 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();
   }
   BufferedInputStream bis = null;
   InputStream is = null;
   OutputStream os = null;
   BufferedOutputStream bos = null;
   try {
      privateFile.createNewFile();
      is = context.getContentResolver().openInputStream(uri);
      bis = new BufferedInputStream(is);
      os = new FileOutputStream(privateFile);
      if (os != null) {
         bos = new BufferedOutputStream(os);
      }
      byte[] buffer = new byte[4096];
      int len = 0;
      while ((len = bis.read(buffer)) != -1) { // 循环从输入流读取 buffer字节
         bos.write(buffer, 0, len); // 将读取的输入流写入到输出流
      }
      if (bis != null) {
         bis.close();
      }
      if (bos != null) {
         bos.close();
      }
   } catch (IOException e) {
      e.printStackTrace();
   }
   return privateFile.getAbsolutePath();
}


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

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

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

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

推荐Android开源项目

项目功能介绍:RxJava2和Retrofit2项目,添加自动管理token功能,添加RxJava2生命周期管理,使用App架构设计是MVP模式和MVVM模式,同时使用组件化,部分代码使用Kotlin,此项目持续维护中。

项目地址:https://gitee.com/urasaki/RxJava2AndRetrofit2