AI智能
改变未来

AndroidQ文件存储适配

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

补充:
安卓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方法,得到图片路径!

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

赞(0) 打赏
未经允许不得转载:爱站程序员基地 » AndroidQ文件存储适配