在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之前的方式,这种需求的数据只能保存到内部存储空间中,而系统的内部存储空间都是有限的,为了让用户能在有限的存储空间中安装更多的应用,所以出现了这种存储方式。个人观点仅供参考 \(^o^)/~。
存储在伪私有目录的数据是不是很不安全?
当然是不安全的,任何应用都有可能访问这些数据,所以一般而言这里只保存那些不是非常敏感的数据,由于这些目录中的数据有可能被用户或者其他应用删除掉,所以在使用这里面的数据的时候要先检测下数据是否存在、有效,如果无效则需要从新下载使用。
这种存储方式适合那些数据呢?
比如一个图片浏览类的应用,用该目录来保存系统图片的缩略图方便提高应用浏览图片的流畅度;或者图书类应用用来保存图书的封面图片等。
如果您的应用运行在旧的设备(低于2.2的版本)上,则您也应该按照上面的建议把文件保存到 /Android/data//files/ 目录下,这样如果用户更新的设备版本后,这些数据就随着应用的卸载而被删除掉了。
遗留问题
当然也有一些比较有争议的数据,比如 IM聊天软件接收到的图片和自定义表情数据 应该保存到SDCard中呢还是上面介绍的 伪私有数据 存储目录中呢? 如果用户卸载了聊天软件,用户是否期望继续保存他们接收到的图片和自定义表情图片呢? 这个问题恐怕针对不同的用户得到的答案是不一样的。这时,我们咋办??
个人建议针对这种没有明确界限的需求,可以在用户第一次使用程序的时候,引导用户去设置他们的这些数据保存到哪里?保存到外部存储空间(应用卸载后数据依然存在);或者保存到应用外部存储目录中(应用卸载后数据自动删除)。 这种做法类似下载一些PC软件的做法,比如一些软件在卸载的时候会填出对话框让用户选择是否保留用户数据。只不过在Android系统中卸载应用目前没办提供该功能,所以只能在应用第一次使用的时候 让用户知道有这么个设置项。
PS:笔者目前最反感的就是一些应用下载了很多小图片到SDCard中,然后打开图库应用会发现里面有很多这种小图片(比如人的头像啦、物品缩略图啦、图书封面图片啦),这种数据就不应该出现在用户的图库应用中去。 如果把这些数据保存到上述的伪私有数据目录中则 就不会出现这种问题。 目前为了避免这种问题,笔者不得不自己在各种目录中创建“.nomedia”文件。相当繁琐啦!

Android如何正确的保存文件的更多相关文章

  1. Android学习笔记_6_保存文件到SDCard

    一.加入访问sdcard的权限 Environment.getExternalStorageState()方法用于获取SDCard的状态,如果手机装有SDCard,并且可以进行读写,那么方法返回的状态 ...

  2. 无废话Android之android下junit测试框架配置、保存文件到手机内存、android下文件访问的权限、保存文件到SD卡、获取SD卡大小、使用SharedPreferences进行数据存储、使用Pull解析器操作XML文件、android下操作sqlite数据库和事务(2)

    1.android下junit测试框架配置 单元测试需要在手机中进行安装测试 (1).在清单文件中manifest节点下配置如下节点 <instrumentation android:name= ...

  3. 【Android Developers Training】 25. 保存文件

    注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...

  4. [android] 保存文件到手机内存

    /*****************2016年5月4日 更新*******************************/ 知乎:Android 没有沙盒保护机制吗,WhatsApp 信息为何可被随 ...

  5. Android Developers:保存文件

    Android使用一个和其它平台基于硬盘文件系统相似的文件系统.这个课程描述了如何和在Android文件系统使用File APIs读和写文件. 一个File对象适用于读或者写从头到尾没用中断的大型数据 ...

  6. android 保存文件的各种目录列表

    一般的,我们可以通过context和Environment来获取要保存文件的目录 ($rootDir) +- /data -> Environment.getDataDirectory() | ...

  7. Android 读取和保存文件(手机内置存储器)

    1:activity_main.xml <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/androi ...

  8. Android学习笔记——保存文件(Saving Files)

              本人邮箱:JohnTsai.Work@gmail.com,欢迎交流讨论.                 欢迎转载,转载请注明网址:http://www.cnblogs.com/J ...

  9. Android -- 保存文件

    背景                                                                                             我们以常见 ...

随机推荐

  1. 浅析@Deprecated

    如果有一个方法你觉得不合适,想要删除,但是别人已经引用了很多次,删除了会对他人的工作产生影响,那该怎么办? 加入@Deprecated注解即可,看代码: @Test public void test1 ...

  2. 以非root权限安装nginx及运行

    本章主要讲如何在无root权限(包含无sudo权限)条件下于centos命令行中安装nginx以及在大于1024的端口(这里用8080)上运行. 1. 安装 两种方式,一是下载预编译好的rpm包安装, ...

  3. 洛谷 P1731 生日蛋糕

    /*洛谷 1731 生日蛋糕 傻傻的-1 T成了傻逼*/ #include<cstdio> #include<iostream> #include<cmath> # ...

  4. php并发处理

       最近某个项目用php生成文件,但是由于文件量太大,单个进程生成需要很长的时间,所以想并发进行处理.    网上查找了下相关的资料,php本身是没有多线程的概念的,那就只能用多进程了,再找资料却是 ...

  5. PHP关闭Notice错误提示

    原文出处 本文章介绍在php中关闭notice错误的一些做法,但是不得不说的是在关掉错误提示时,那么程序代码中大量的notice级别错误是否会造成PHP性能下降 PHP Notice: Undefin ...

  6. 28、Jquery 页面效果

    Jquery动画函数分为三类 基本动画函数 基本动画函数包括show(显示).hide(隐藏).toggle(切换)这三个函数. 例子所需html <input type="butto ...

  7. CS0016: 未能写入输出文件“c:\Windows\Microsoft.NET\Framework64\v4.0.30319\Temporary ASP.NET Files\root\d29b5393\123c3a1c\App_Code.odl3w4o6.dll”--“拒绝访问。 ”

    IIS部署网站或者Webservice时,出现以下问题: CS0016: 未能写入输出文件“c:\Windows\Microsoft.NET\Framework64\v4.0.30319\Tempor ...

  8. font awesome icon

    http://fontawesome.io/icons/ http://www.bootstrapicons.com/

  9. insert一句话实现插入一条数据并且返回这条数据的某列

    insert into [table] output inserted.columnName values();

  10. js跨浏览器事件对象、事件处理程序

    项目中有时候会不用jquery这么好用的框架,需要自己封装一些事件对象和事件处理程序,像封装AJAX那样:这里面考虑最多的还是浏览器的兼容问题,原生js封装如下:var EventUtil={ //节 ...