Android WRITE_EXTERNAL_STORAGE权限

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使用相应权限,动态请求相应权限就总是会失败。

    原文作者:BrightVan
    原文地址: https://www.jianshu.com/p/f0b0924b0121
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞