Android6.0权限适配之WRITE_EXTERNAL_STORAGE(SD卡写入)

前一篇博客中介绍了Android6.0运行时权限,最近遇到这么一个情况,就是一个App以前都是在SD卡根目录直接新建了一个XXX/image/目录,来保存图片缓存的,但是如果适配到Android6.0,我们就需要弹出对话框给用户,来申请WRITE_EXTERNAL_STORAGE权限,如果仅仅是缓存图片为了提高加载速度,对于一个小白用户来讲,好像并不是什么值得让他授权的理由。。。

下面记录一下我是怎么处理的,其实这次处理也不能叫做Android6.0权限适配了,不过对于WRITE_EXTERNAL_STORAGE这个权限而言,的确有一些需要注意的地方(坑)使我们以前没有关心的。

首先,App在手机上保存文件或者缓存数据时,我认为应该遵守以下几点:

1.不要随意占用用户的内置存储。

2.不要随意在SD卡上新建目录,应该放置自己应用包名对应的扩展存储目录下,卸载App时可以被自动清除。

3.对占用的磁盘空间有上限,并按照一定的策略进行清除。

Android下有哪些文件目录

在Android系统中,根据调用的系统API接口,有3种目录可以给我们写入文件:

1.应用私有存储(内置存储)

获取方式:

Context.getFileDir():获取内置存储下的文件目录,可以用来保存不能公开给其他应用的一些敏感数据如用户个人信息

Context.getCacheDir():获取内置存储下的缓存目录,可以用来保存一些缓存文件如图片,当内置存储的空间不足时将系统自动被清除(然而具体多大,清除时的策略我也没查到。。)

绝对路径:

Context.getFileDir():/data/data/应用包名/files/

Context.getCacheDir():/data/data/应用包名/cache/

写权限:不需要申请

这是手机的内置存储,没有root的过的手机是无法用文件管理器之类的工具查看的。而且这些数据也会随着用户卸载App而被一起删除。这两个目录其实就对应着设置->应用->你的App->存储空间下面的清除数据和清楚缓存,如下图所示。

2.应用扩展存储(SD卡)

获取方式:

Context.getExternalFilesDir():获取SD卡上的文件目录

Context.getExternalCacheDir():获取SD卡上的缓存目录

绝对路径:

Context.getExternalFilesDir():SDCard/Android/data/应用包名/files/

Context.getExternalCacheDir():SDCard/Android/data/应用包名/cache/

写权限:

API < 19:需要申请

API >= 19:不需要申请

既然是SD卡上的目录,那么是可以被其他的应用读取到的,所以这个目录下,不应该存放用户的敏感信息。同上面一样的,这里的文件会随着App卸载而被删除,也可以由用户手动在设置界面里面清除。

3.公共存储(SD卡)

获取方式:Environment.getExternalStorageDirectory()

绝对路径:SDCard/你设置的文件夹名字/

写权限:需要申请

如果我们的App需要存储一些公共的文件,甚至希望下载下来的文件即使在我们的App被删除之后,还可以被其他App使用,那么就可以使用这个目录。这个目录是始终需要申请SD写入权限的。

Android6.0下应该把文件放到哪里?

有了前一节的介绍,其实很清楚了,根据最开始提到的规则,其实如果仅仅是做了简单的图片缓存工作,那么我们应该把图片缓存放到/data/data/应用包名/cache/或者SDCard/Android/data/应用包名/cache/,因为在6.0系统(API > 23)时,不需要申请权限就可以向这两个目录写入文件。而且/data/data/应用包名/cache/目录,是内置存储的应用私有缓存目录,在系统空间不够时还会被自动清除,对于图片缓存来讲也是一个不错的管理策略,不过谷歌建议我们最好还是自己实现缓存清除管理,例如用DiskLruCache。

实际上我们可以在API >= 19(不一定非要大于23)时,就可以在不需要申请权限的情况下把文件放到这两个目录了。如果开发的时候足够规范,即使在API < 19时,我们申请到写入权限后,我们也应该手动创建和前面相同的目录,使得应用存储数据目录统一化。

Last,最后还有个坑!

好了,是不是现在不用SD卡上创建的目录XXX/image/,直接改为改为SDCard/Android/data/应用包名/cache/image/就OK了?还真不完全是这样的。。。

???纳尼????

通常我们开发App时会设置targetSDKVersion=23时,并同时向前兼容,还会设置minSdkVersion=14表示支持的最低系统版本是Android4.0(API = 14)。也就是说我们的build.gradle一般长这样:

android {

compileSdkVersion 23

buildToolsVersion"23.0.2"

...

defaultConfig {

applicationId"xxx.xxx"

minSdkVersion 14

targetSdkVersion 23

versionCode 1

versionName"1.0"

}

...

}

但是前面我们说过,通过Context.getExternalCacheDir()接口获取应用扩展存储目录时,只有在API >= 19时才不需要申请权限。也就是说如果是上面这种兼容到API 14的应用,还是需要在AndroidManifest.xml中注册WRITE_EXTERNAL_STORAGE权限的。

前一篇博客Android6.0运行时权限简介知道,如果在AndroidManifest.xml文件里注册过WRITE_EXTERNAL_STORAGE,当App运行在一台6.0的设备时,即使你的App全程都没有调用requestPermissons来申请权限,用户还是可以在Android6.0系统上进入设置->应用->你的App->权限里面,取消存储空间这一个权限。记住是运行在6.0系统的机器上,这是关键,因为低于6.0的系统根本没有这个设置。

如下图所示,只要在manifest里面注册了,就可以动态取消之!

此时会发生什么???此时你的图片在6.0机器上也就没法缓存喽。。/(ㄒoㄒ)/~~

为啥啊?6.0机器上,我不是不需要申请权限就可以获得写入SDCard/Android/data/应用包名/cache/目录吗?实际测试时发现,当用户取消了权限之后,SDK接口中与File相关的API全部都返回空了,于是我们就没法写文件了。

其实我们还需要做的是:

将AndroidManifest.xml文件中的

改为

android:maxSdkVersion="18"/>

表示只在API <= 18时,才申请WRITE_EXTERNAL_STORAGE权限。这样用户就无法在Android6.0系统的设置下面看到存储空间权限的开关,当然也就无法关闭它了,如下图所示。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,036评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,046评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,411评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,622评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,661评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,521评论 1 304
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,288评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,200评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,644评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,837评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,953评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,673评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,281评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,889评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,011评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,119评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,901评论 2 355

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,130评论 25 707
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,656评论 18 139
  • open.weixin.qq.com/connect/oauth2/authorize (引用的文章地址) 方便以...
    兀兀沙弥阅读 2,214评论 1 1
  • 参考链接:leanote.com/blog/post/561658f938f41126b2000298 本文描述了...
    灬黑客灬阅读 414评论 0 0
  • 今天的围棋课用时1小时,课堂第一章、先拼图,再温习实空和点目;课堂第二章、讲故事《唐王李世民与虬髯客下围棋》;故事...
    南山围棋阅读 354评论 0 0