在Android官方开发文档中有一篇文档来介绍如何保存应用的数据, 但笔者用过很多程序(从知名的到不知名的)处理的都不是很完美,或者没有按照Android开发团队建议的方式去保存他们应用的数据。当一些试用过的软件卸载后,在SDCard中还保留了这些软件的文件目录,时间一长就有很多目录需要手工去清理,并且很难确认那些目录是仍然在使用的那些目录已经没用了,给用户带来困扰。这里我们来讨论下该如何正确的保存应用的数据。
首先数据分为两种:和应用相关的数据;和应用无关的数据。这里分别介绍:
应用无关的数据是那些用户比较关心的数据,不管您的应用在不在用户设备上,这些数据用户都希望保留,这些数据包含:用相机拍摄的照片、用浏览器(下载工具)下载的文件、用户制作的个性铃声等。
假设您开发了一个照相应用(例如: 360相机),用户用您的应用拍摄并处理过的照片就属于应用无关的数据,如果用户把您的应用给卸载了,用户还是会期望仍然保留他们拍摄并处理过的照片。这里的照片按照Android官方的建议应该保存到 DIRECTORY_PICTURES 目录中,该目录通过Environment.getExternalStoragePublicDirectory(String type)来获取,您可以在该目录下创建一个以您的程序命名的目录来保存数据。其他支持的目录列表请参考这里。
应用相关的数据: 这种数据只和您的应用相关,如果您的应用被用户删除了这些数据也没有理由还存在用户设备中。这种数据包含:数据库文件、属性配置文件、应用的缓存文件等。这种数据可以有很多种保存方式,详情参考这里。
这里我们只讨论在API 8(Android 2.2)中新引入了一种保存到外部存储空间的伪私有数据API, 通过函数getExternalFilesDir() 来获取该路径,同样可以设置获取各种类型数据的参数,例如 DIRECTORY_MUSIC 和 DIRECTORY_RINGTONES (如果参数为null则返回您应用数据的跟目录)。比如一个应用的包名为 org.goodev.test 的应用,通过函数getExternalFilesDir(Environment.DIRECTORY_MOVIES)获取到的文件路径为 /storage/sdcard0/Android/data/org.goodev.test/files/Movies。
细心的读者已经发现,该数据目录为SDCard (外部存储设备,有可能是内置SDCard 比如 Nexus S) 中名字为 Android 的根目录下,该目录下有个用来保存应用数据的 data 目录,在这个 data 目录中保存了设备中各个应用的数据,依包名来命名,如果设置参数为null则返回的是跟目录:/storage/sdcard0/Android/data/org.goodev.test/files。
为啥这个目录我们称之为伪私有数据目录呢? 因为在2.2以上的系统中,当您的应用被用户卸载的时候,保存在这个目录下的数据也会被系统删除;并且在默认情况下多媒体扫描器不会扫描该目录下的图片、 MP3等多媒体文件– 从这个角度看这个目录是应用的私有数据目录。 由于该目录存在于外部存储空间中,任何其他具有读写外部存储空间权限的应用都可以访问您的应用数据 — 从这个角度看这些数据又不是私有的。 所以我们就称之为 伪私有数据。
那么为什么Android在2.2中引入这种伪私有数据保存方式呢? 这样应用的数据不是很不安全吗? 这种存储方式适合保存那些数据呢? 下面我们来逐个分析下这些问题:
为何引入这种存储方式?
有些Android应用在使用过程中,可能会产生很多需要保存的文件数据,而这些数据需要在应用卸载的时候被删除掉,按照2.2之前的方式,这种 需求的数据只能保存到内部存储空间中,而系统的内部存储空间都是有限的,为了让用户能在有限的存储空间中安装更多的应用,所以出现了这种存储方式。
存储在伪私有目录的数据是不是很不安全?
当然是不安全的,任何应用都有可能访问这些数据,所以一般而言这里只保存那些不是非常敏感的数据,由于这些目录中的数据有可能被用户或者其他应用删除掉,所以在使用这里面的数据的时候要先检测下数据是否存在、有效,如果无效则需要从新下载使用。
这种存储方式适合那些数据呢?
比如一个图片浏览类的应用,用该目录来保存系统图片的缩略图方便提高应用浏览图片的流畅度;或者图书类应用用来保存图书的封面图片等。
如果您的应用运行在旧的设备(低于2.2的版本)上,则您也应该按照上面的建议把文件保存到 /Android/data//files/ 目录下,这样如果用户更新的设备版本后,这些数据就随着应用的卸载而被删除掉了。
遗留问题
当然也有一些比较有争议的数据,比如 IM聊天软件接收到的图片和自定义表情数据 应该保存到SDCard中呢还是上面介绍的 伪私有数据 存储目录中呢? 如果用户卸载了聊天软件,用户是否期望继续保存他们接收到的图片和自定义表情图片呢? 这个问题恐怕针对不同的用户得到的答案是不一样的。这时,我们咋办??
个人建议针对这种没有明确界限的需求,可以在用户第一次使用程序的时候,引导用户去设置他们的这些数据保存到哪里?保存到外部存储空间(应用卸载 后数据依然存在);或者保存到应用外部存储目录中(应用卸载后数据自动删除)。 这种做法类似下载一些PC软件的做法,比如一些软件在卸载的时候会填出对话框让用户选择是否保留用户数据。只不过在Android系统中卸载应用目前没办 提供该功能,所以只能在应用第一次使用的时候 让用户知道有这么个设置项。
PS:笔者目前最反感的就是一些应用下载了很多小图片到SDCard中,然后打开图库应用会发现里面有很多这种小图片(比如人的头像啦、物品缩略 图啦、图书封面图片啦),这种数据就不应该出现在用户的图库应用中去。 如果把这些数据保存到上述的伪私有数据目录中则 就不会出现这种问题。 目前为了避免这种问题,笔者不得不自己在各种目录中创建“.nomedia”文件。相当繁琐啦!