1、问题由来
注:严格来说,下面的<uses-permission ..../>统一用“使用权限”指代,<permission .../>才应该叫声明权限。
写的好好的代码,忽然之前的某个功能(要在外部存储的自建目录写文件)不能正常使用了!可是我根本就没改过相关的任何代码啊!
表现就是动态请求“WRITE_EXTERNAL_STORAGE”权限总是失败。相机权限请求是正常的,和SVN上对比也找不到问题之所在(之前的版本是没问题的),总之非常迷,无语!遇到这种问题是最烦的。而经验也告诉我不是动态权限申请的问题。肯定是其他不直接关联的细节错误。
果不其然,在一个依赖module的Manifest文件中我找到了原因,当时感觉就是卧槽。。。
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
android:maxSdkVersion="18"/>
关键就是“android:maxSdkVersion="18"”导致了错误。上面的权限使用和主app module的有冲突,而且很明显,被覆盖了。导致在4.4以上的设备注册表中没有使用这个权限,进而导致动态权限申请也一直失败!
2、“WRITE_EXTERNAL_STORAGE”权限
根据安卓开发者文档的说明:从API Lever19(Android4.4)开始,如果在外部存储的应用“缓存目录”读写数据是不需要添加该权限的,目录路径如下(后面提到的“缓存目录”皆指下面的目录):
Context.getExternalFilesDir():SDCard/Android/data/应用包名/files/
Context.getExternalCacheDir():SDCard/Android/data/应用包名/cache/
所以,文档就建议如果权限使用设置成下面的方式:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
android:maxSdkVersion="18"/>
但仅适用于你在自己应用的“缓存目录”下读写的情况。如果要在其他目录就要去掉“android:maxSdkVersion="18"”。
“缓存目录”很明显和本应用相关,应用删除后,相应的也会删除这些目录的文件,如果做些缓存,放在这里自然没问题。如果要长期保存(即使应用卸载了)就要在外部存储新建目录:
Environment.getExternalStorageDirectory():SDCard/你设置的文件夹名字/
那官方为什么还建议设置“android:maxSdkVersion="18"”呢?
6.0开始安卓系统引入了更严格的权限管理机制,你在Manifest声明的敏感权限都可以在设置中手动关闭的。如果用户手动关闭了,相应的文件写操作的接口都不能正常工作,即使是在自己应用的“缓存目录”。但加了“android:maxSdkVersion="18"”,在安卓4.4以上设备就相当于没有使用相应权限,权限设置项里就没有相应项,也就不存在被用户手动关闭权限的情况。
3、总结
(1)、安卓4.4以上,如果只在“缓存目录”读写数据,就不需要使用“WRITE_EXTERNAL_STORAGE”权限。但应用要兼容到安卓4.4以下还是要使用的,也就是下面的形式:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
android:maxSdkVersion="18"/>
(2)、如果在公共存储读写数据就必须使用“WRITE_EXTERNAL_STORAGE”权限,也就要去掉“ android:maxSdkVersion="18"”。安卓6.0以上系统,用户可手动关闭该权限。
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
(3)、不在Manifest使用相应权限,动态请求相应权限就总是会失败。