Android的学习路上...
- 测试设备:vivo X90s
- 安卓版本: Android13
- 开发环境:AndroidStudio Flamingo
- SDK:33
最近我在Android13的环境下尝试写一个文件选择器,以便日后的开发使用。但是我们知道,从Android13 (API33) 开始,外部存储权限发生了变化,要想读取外部存储文件,使用原来的权限注册方式已经失效了。
<!--以下方式在安卓13下已失效--> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
测试一下:我在test文件夹内有4个文件和1个文件夹。
左图是系统自带的文件选择器,右图是我开发的文件选择器(我是以系统的选择器作为参照的,因此样子长得很像😄)。此时我的文件选择器只注册了 WRITE/READ_EXTERNAL_STORAGE 权限。可以看到,这种方式在Android13中是获取不到文件的,只能获取文件夹。
这是因为Android13细化了外部存储权限,在我的理解里,Android13把文件分为了媒体文件和非媒体文件。媒体文件包括照片、视频和音频,他们需要各自的权限才能获取。权限声明如下所示(在AndroidManifest.xml中声明后还需动态获取权限):
<!--照片--> <uses-permission android:name="android.permission.READ_MEDIA_IMAGES" /> <!--视频--> <uses-permission android:name="android.permission.READ_MEDIA_VIDEO" /> <!--音频--> <uses-permission android:name="android.permission.READ_MEDIA_AUDIO" />
在上图中,picture.png属于照片文件,video.mp4属于视频文件,music.mp3属于音频文件。而text.pdf就属于非媒体文件了。若现在赋予我的文件选择器媒体文件权限,如下图所示,此时就能看到:picture.png、video.mp4、music.mp3就能被获取了。
那除了媒体文件外,其他的比如PDF文件、DOCX文件、TXT文件等等这些非媒体文件应该如何获取呢?其中之一可行的办法就是使用管理所有文件权限。
<!--管理所有文件权限--> <uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" tools:ignore="ScopedStorage" />
使用此权限时,需要现在AndroidManifest.xml中声明,但不需动态获取权限。而是要跳转到设置页面使用户手动开启权限,跳转代码如下:
// 跳转至设置页面,让用户手动开启 public void setting(Activity activity) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) { if (!Environment.isExternalStorageManager()) { Intent intent = new Intent( Settings.ACTION_MANAGE_APP_ALL_FILES_ACCESS_PERMISSION); intent.setData(Uri.parse("package:" + activity.getPackageName())); activity.startActivity(intent); } } }
设置页面如下左图。手动开启了权限后,就可以获取到其他文件了,如下右图所示。
开启此权限值得注意的是:
如果允许了此权限,那么即使应用无“照片与视频”、“音乐与音频”权限,仍可访问、写入、修改和删除所有文件。
也就是说,如果开启了这个权限,就可以获取所有的文件了(包括媒体文件和非媒体文件),那么媒体文件的权限就算不申请也可以获得权限。
总结:
1. 如果业务需求只获取媒体文件,那么就 AndroidManifest.xml 注册以下权限并动态申请权限。
<!--照片--> <uses-permission android:name="android.permission.READ_MEDIA_IMAGES" /> <!--视频--> <uses-permission android:name="android.permission.READ_MEDIA_VIDEO" /> <!--音频--> <uses-permission android:name="android.permission.READ_MEDIA_AUDIO" />
2. 如果业务需求获取非媒体文件或所有文件,那么就在 AndroidManifest.xml 注册以下权限并跳转至设置页面使用户手动开启权限。
<!--管理所有文件权限--> <uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" tools:ignore="ScopedStorage" />
// 跳转至设置页面,让用户手动开启 public void setting(Activity activity) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) { if (!Environment.isExternalStorageManager()) { Intent intent = new Intent( Settings.ACTION_MANAGE_APP_ALL_FILES_ACCESS_PERMISSION); intent.setData(Uri.parse("package:" + activity.getPackageName())); activity.startActivity(intent); } } }
这是我在开发中尝试的一种可行的方案,如果大家有其他可行的方案欢迎留言。如文章有错误之处,强烈欢迎指正,在下萌新需要多多学习和积累,谢谢啦~