可能遇到的问题

android系统自身自带有存储,另外也可以通过sd卡来扩充存储空间。前者好比pc中的硬盘,后者好移动硬盘。 前者空间较小,后者空间大,但后者不一定可用。 开发应用,处理本地数据存取时,可能会遇到这些问题:

  1. 需要判断sd卡是否可用: 占用过多机身内部存储,容易招致用户反感,优先将数据存放于sd卡;
  2. 应用数据存放路径,同其他应用应该保持一致,应用卸载时,清除数据:

    • 标新立异在sd卡根目录建一个目录,招致用户反感
    • 用户卸载应用后,残留目录或者数据在用户机器上,招致用户反感
  3. 需要判断两者的可用空间: sd卡存在时,可用空间反而小于机身内部存储,这时应该选用机身存储;

  4. 数据安全性,本应用数据不愿意被其他应用读写;
  5. 图片缓存等,不应该被扫描加入到用户相册等媒体库中去。

基本操作

  1. 使用外部存储,需要的权限,在 AndoridManifest.xml 中:

    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

    从API 19 / Andorid 4.4 / KITKAT开始,不再需要显式声明这两个权限,除非要读写其他应用的应用数据( $appDataDir )

  2. 判断sd卡可用:

    /**
    * Check if the primary "external" storage device is available.
    *
    * @return
    */
    public static boolean hasSDCardMounted() {
    String state = Environment.getExternalStorageState();
    if (state != null && state.equals(Environment.MEDIA_MOUNTED)) {
    return true;
    } else {
    return false;
    }
    }

存储的用量情况

  • 根据系统用户不同,所能占用的存储空间大小也有不同

    在API level 9及其以上时, File 对象的 getFreeSpace() 方法获取系统root用户可用空间;

    getUsableSpace() 取非root用户可用空间

  • 当有多个存储可用时获取磁盘用量,根据当前系统情况选用合适的存储。

  • 根据系统存储用量,合理设定app所用的空间大小;运行时,也可做动态调整。

  • 在API level 9及其以上的系统,可直接调用 File 对象的相关方法,以下需自行计算:

    @TargetApi(VERSION_CODES.GINGERBREAD)
    public static long getUsableSpace(File path) {
    if (path == null) {
    return -1;
    }
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.GINGERBREAD) {
    return path.getUsableSpace();
    } else {
    if (!path.exists()) {
    return 0;
    } else {
    final StatFs stats = new StatFs(path.getPath());
    return (long) stats.getBlockSize() * (long) stats.getAvailableBlocks();
    }
    }
    }

路径的规律

一般地,通过 Context  和  Environment 相关的方法获取文件存取的路径。

通过这两个类可获取各种路径,如图:

    ($rootDir)
+- /data -> Environment.getDataDirectory()
| |
| | ($appDataDir)
| +- data/com.srain.cube.sample
| |
| | ($filesDir)
| +- files -> Context.getFilesDir() / Context.getFileStreamPath("")
| | |
| | +- file1 -> Context.getFileStreamPath("file1")
| | ($cacheDir)
| +- cache -> Context.getCacheDir()
| |
| +- app_$name ->(Context.getDir(String name, int mode)
|
| ($rootDir)
+- /storage/sdcard0 -> Environment.getExternalStorageDirectory()
| / Environment.getExternalStoragePublicDirectory("")
|
+- dir1 -> Environment.getExternalStoragePublicDirectory("dir1")
|
| ($appDataDir)
+- Andorid/data/com.srain.cube.sample
|
| ($filesDir)
+- files -> Context.getExternalFilesDir("")
| |
| +- file1 -> Context.getExternalFilesDir("file1")
| +- Music -> Context.getExternalFilesDir(Environment.Music);
| +- Picture -> ... Environment.Picture
| +- ...
|
| ($cacheDir)
+- cache -> Context.getExternalCacheDir()
|
+- ???

各个路径的特性

下面介绍这些路径的特性以及使用中需要注意的细节:

  1. 根目录( $rootDir ):

    • 内部存储路径:  /data , 通过 Environment.getDataDirectory()  获取
    • 外部存储路径:  /storage/sdcard0  (也有类似 /mnt/ 这样的),通过 Environment.getExternalStorageDirectory() 获取

      示例:

        Environment.getDataDirectory():
      /data Environment.getExternalStorageDirectory():
      /storage/sdcard0

  2. 应用数据目录( $appDataDir )
    • 内部储存:  $appDataDir = $rootDir/data/$packageName ,
    • 外部存储:  $appDataDir = $rootDir/Andorid/data/$packageName

    在这些目录下的数据,在app卸载之后,会被系统删除,我们应将应用的数据放于这两个目录中。


  3. 外部存储中,公开的数据目录。 这些目录将不会随着应用的删除而被系统删除,请斟酌使用:

     Environment.getExternalStorageDirectory():
    /storage/sdcard0 // 同 $rootDir
    Environment.getExternalStoragePublicDirectory(""):
    /storage/sdcard0 Environment.getExternalStoragePublicDirectory("folder1"):
    /storage/sdcard0/folder1

  4. 应用数据目录下的目录

    一般的在$appDataDir下,会有两个目录 :

    1. 数据缓存: $cacheDir = $appDataDir/cache :

      • 内部存储: Context.getCacheDir() , 机身内存不足时,文件会被删除
      • 外部存储: Context.getExternalCacheDir()

        外部存储没有实时监控,当空间不足时,文件不会实时被删除,可能返回空对象

        示例:

          Context.getCacheDir():
        /data/data/com.srain.cube.sample/cache Context.getExternalCacheDir():
        /storage/sdcard0/Android/data/com.srain.cube.sample/cache
    2. 文件目录  $filesDir = $appDataDir/files :
      • 内部存储:通过 Context.getFilesDir()  获取

        Context.getFileStreamPath(String name) 返回以 name 为文件名的文件对象, name 为空,则返回  $filesDir  本身

        示例:

          Context.getFilesDir():
        /data/data/com.srain.cube.sample/files Context.getFileStreamPath(""):
        /data/data/com.srain.cube.sample/files Context.getFileStreamPath("file1"):
        /data/data/com.srain.cube.sample/files/file1
      • 外部存储:通过 Context.getExternalFilesDir(String type) ,  type 为空字符串时获取.

        type 系统指定了几种类型:

          Environment.DIRECTORY_MUSIC
        Environment.DIRECTORY_PICTURES
        ...

        示例:

          Context.getExternalCacheDirs():
        /storage/sdcard0/Android/data/com.srain.cube.sample/files Context.getExternalFilesDir(Environment.DIRECTORY_MUSIC)
        /storage/sdcard0/Android/data/com.srain.cube.sample/files/Music

    3. $cacheDir / $filesDir  安全性

      在内部存储中, $cacheDir ,  $filesDir 是app安全的,其他应用无法读取本应用的数据,而外部存储则不是。

      在外部存储中,这两个文件夹其他应用程序也可访问。

      在外部存储中, $filesDir 中的媒体文件,不会被当做媒体扫描出来,加到媒体库中。


    4. $cacheDir / $filesDir  同级目录

      在内部存储中:通过  Context.getDir(String name, int mode) 可获取和  $filesDir  /  $cacheDir 同级的目录

      目录的命名规则为  app_ + name , 通过mode可控制此目录为app私有还是其他app可读写。

      示例:

      Context.getDir("dir1", MODE_PRIVATE):
      Context.getDir: /data/data/com.srain.cube.sample/app_dir1

    5. 特别注意, 对于外部存储,获取 $cacheDir  或者  $filesDir 及其下的路径

      在API level 8 以下,或者空间不足,相关的方法获路径为空时,需要自己构造。

      @TargetApi(VERSION_CODES.FROYO)
      public static File getExternalCacheDir(Context context) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.FROYO)) {
      File path = context.getExternalCacheDir(); // In some case, even the sd card is mounted,
      // getExternalCacheDir will return null
      // may be it is nearly full.
      if (path != null) {
      return path;
      }
      } // Before Froyo or the path is null,
      // we need to construct the external cache folder ourselves
      final String cacheDir = "/Android/data/" + context.getPackageName() + "/cache/";
      return new File(Environment.getExternalStorageDirectory().getPath() + cacheDir);
      }

Android文件存储使用参考的更多相关文章

  1. Android 文件存储浅析

    最近做的一个需求和文件存储有关系.由于之前没有系统梳理过,对文件存储方面的知识一直很懵懂.趁着周末有时间,赶紧梳理一波. 这首从网上找到的一张图,很好的概括了外部存储和内部存储. 下面我们再来具体介绍 ...

  2. android文件存储位置切换

    最近有个需求,助手的google卫星地图和OpenCycleMap下载的离线地图数据,要能够在内置存储和外置存储空间之间切换,因为离线瓦片数据非常大,很多户外用户希望将这些文件存储在外置TF卡上,不占 ...

  3. Android文件存储

    文件存储是Android中最基本的一种数据存储方式,它不读存储的内容进行任何的格式化处理,所有数据原封不动的保存在文件之中.如果想用文件存储的方式保存一些较为复杂的数据,就需要定义一套自己的格式规范, ...

  4. android: 文件存储

    数据持久化就是指将那些内存中的瞬时数据保存到存储设备中,保证即使在手机或电脑 关机的情况下,这些数据仍然不会丢失.保存在内存中的数据是处于瞬时状态的,而保存在 存储设备中的数据是处于持久状态的,持久化 ...

  5. 转:Android文件存储路径getFilesDir()与getExternalFilesDir的区别

    作为一个开发者,我们经常需要通过缓存一些文件到SD卡中,常见的方式就是,通过: File sdCard = Environment.getExternalStorageDirectory(); 获取S ...

  6. android 文件存储&SharedPreferences

    一.文件存储 文件存储主要是I/O流的操作,没什么好说的,需要注意的是保存文件的各个目录. 下面为常用的目录: public static File getInFileDir(Context cont ...

  7. 程序员带你学习安卓开发系列-Android文件存储

    这是程序员带你学习安卓开发系列教程.本文章致力于面向对象程序员可以快速学习开发安卓技术. 上篇文章:.Net程序员快速学习安卓开发-布局和点击事件的写法 主要讲解了布局和点击事件的写法. 上篇文章补充 ...

  8. Android 文件存储 和 权限管理

    转载请标明出处: :http://blog.csdn.net/huaiyiheyuan/article/details/52473984 android SD卡主要有两种存储方式 Internal . ...

  9. Android数据存储之Android 6.0运行时权限下文件存储的思考

    前言: 在我们做App开发的过程中基本上都会用到文件存储,所以文件存储对于我们来说是相当熟悉了,不过自从Android 6.0发布之后,基于运行时权限机制访问外置sdcard是需要动态申请权限,所以以 ...

随机推荐

  1. error: C2664: “zajiao::zajiao(const zajiao &)”: 无法将参数 1 从“const char [12]”转换为“char *”

    原本打算在QT用一个字符串"ABCDEF12345"作为类zajiao的构造函数的参数,用来创建类zajiao的对象zajiao1. zajiao zajiao1("AB ...

  2. hdu 2604 Queuing(矩阵快速幂乘法)

    Problem Description Queues and Priority Queues are data structures which are known to most computer ...

  3. Java可视化编程,基于布局管理器的UI设计

    在<事件驱动模型>讲述了如何将用户与功能实现代码联系到一起.怎么样便于用户理解和符合用户的使用习惯? 本篇还是就此问题作分析,站在用户角度上分析UI各组件倒底该如何设计呈现. 优秀的UI会 ...

  4. DOM注意事项(八):JavaScript操作环境和垃圾收集

    一.运行环境 在约JavaScript对象或this当指向问题,念就是运行环境.即上下文环境.运行环境在JavaScript是一个非常重要的概念.由于它定义了变量或函数有权訪问的其他数据,决定了它们各 ...

  5. 依赖注入及AOP简述(十三)——AOP应用举例(完结) .

    2.     AOP应用举例 在一般的应用程序开发中,有一些典型的AOP应用,使得开发者可以专注于业务逻辑本身,而不是与之完全无关的一些“方面”. l        首先就是关于前面介绍过的日志输出类 ...

  6. SharePoint2010 Form验证配置流程

    1.修改管理中心的Web.config文件,位置:C:\inetpub\wwwroot\wss\VirtualDirectories\42903 2.修改应用程序的Web.config文件,位置:C: ...

  7. Spire PDF for .NET 在ASP.NET中的使用 ---- 并非那么“美好”,有些挫折!

    笔者注:看此文前,请您先看一下上一篇文章吧. 昨天的时候,我测试了一下Spire PDF在WinForm程序中的应用,可以说用起来很简单(请忽略效率问题,没有进行测试).不过在互联网如此发达的今天,适 ...

  8. MSSQL查询连接数

    SELECT * FROM [Master].[dbo].[SYSPROCESSES] WHERE [DBID] IN ( SELECT [DBID] FROM [Master].[dbo].[SYS ...

  9. CRM后期修改实体,新增货币类型字段 需要注意的问题

    货币类型字段新增 需要处理历史数据 否则编辑会报错 提示如果货币字段中存在值,则需要指定币种,请选择币种,然后重试 编辑时货币字段不显示¥符号.新增正常.第一次编辑提示错误保存后再编辑也正常.不是JS ...

  10. 安装Eclipse Html Editor

    最近在eclipse中开发android项目,用到了jquery mobile框架,则会涉及到新建html文件,发现eclipse不自带新建html文件的插件,必须得新建一个其他形式的文件,譬如xml ...