在项目开发中经常需要存储一些数据在本地,在存储过程中往往需要设置存储地址,关于存储地址虽然经常用到,但是有些朋友依旧懵懂不知,索性简单记录一下~

Hint:均为自己所想,无法保证是否完全正确


  • 存储区别
  • 物理角度
  • 逻辑角度
  • 分区角度
  • 存储特点
  • 目录特点
  • 存储地址(路径)
  • 兴趣扩展


so look here

  • 关于存储方式主要有五种方式,分别为文件存储sp存储、ContentProvider、数据库存储、网络缓存
  • 关于存储分类主要分为 内部存储、外部存储

内部存储与外部存储的使用场景,主要看项目需求和自身对项目的考虑,如果有的数据需要长久保存,那么有的开发人员会放在外部存储中;但是我自己的话,更多的还是使用内部存储,防止为手机产生无效垃圾~

在补充内存方面知识的时候,看到了一篇介绍Android文件系统的目录结构blog,特此借用一下这张图

正式介绍之前呢,我想说明一下常用的内存路径一般有data/dataAndroid/data 俩种路径,除了私有、共有的概念外,还有哪些需要了解的?都需要通过下方的知识来答疑解惑了 ~

  • data/data 的安装目录,需要获取root权限才能查看, 如:百度地图的安装路径是/data/data/com.baidu.com/ (获得内部存储根目录地址Environment.getDataDirectory())
  • system 存放系统应用的apk文件,即手机厂商预安装应用的apk文件 (手机厂商只需把需要预安装的apk放在该节点的相应路径下,android系统就会自己解压并安装该apk - Environment.getRootDirectory()获取目录节点
  • storage该节点是内置存储卡和外置SD卡的挂载点,/storage/emulated/0/是内置存储卡挂载点, /storage/sdcard1是外置SD卡挂载点(不同的设备挂载节点不一样,有些设备可能会挂载到/mnt/节点)

看完上面的图有没有越来越乱的感觉?实话说:我确实还是有点一知半解,不过在结合另一篇 blog 后,稍有解惑 ~

通过下图我们可以看出Andorid4.4之后并不是没有sd卡了,而是手机内置了外部存储卡,不在需要用户通过sd卡槽插入sd卡了;那么通过这个逻辑我们去看待上方的图时,针对storage(内存)节点我们可以将其看做是内置的外部存储,而关于data节点我们可以看做是手机自身的内部存储 ~

Android12 申请外部储存权限 android内部存储和外部存储_Android

如看完整篇依旧存在一些疑惑的话,也可以去这篇blog再次进阶一些,我感觉写的也蛮细致的 ~

存储区别

关于内部存储与外部存储的区别,其实是一个很模糊又有点抽象的概念,我个人仅从以下的几个角度来阐释这个问题

物理角度

从物理角度看待这个问题(这个区别点,站不稳 - -

  • 手机自身存储:内部存储
  • SD卡:外部存储
逻辑角度

通过内存节点在区分内部存储与外部存储

  • data节点:内部存储
  • mnt或storage节点:外部存储
分区角度

通过 Environment.getExternalStorageDirectory()获取到内置外部存储地址,内部分为私有目录和公共目录(这个区别点,站不稳

  • 内部存储:私有目录
  • 外部存储:公共目录
存储特点
  • 内部存储:存储空间一般较小,主要存储一些系统数据与系统应用程序数据
  • 外部存储:存储空间较大,一般我将app的数据放置于对应包名的目录下
目录特点
  • 私有目录:存储资源会随着app的卸载而自动删除,其他app无法访问当前app内部存储的数据,除了用非法手段或者我们主动暴露
  • 公共目录:存储资源不会自动删除, 除非用户手动删除, 否则会一直存在, 换句话说就是垃圾… ,不过如果需要其他app使用我们的数据,是有人会这么操作的,不过挺危险的
存储地址(路径)

内部存储(私有目录)

关于内部存储路径有俩种获取方式,主用通过Context 来获取,Google官方建议把数据存储在 /sdcard/Android/data/包名/(外部存储) 下

/data/data/包名/(需要root权限)
// 返回值:/data/data/包名/files
context.getFilesDir();
// 返回值:/data/data/包名/cache  
context.getCacheDir();
  • 使用方式
Log.e("FileTag", "getFilesDir())-File path:" + getFilesDir());
 Log.e("FileTag", "getCacheDir())-File path:" + getCacheDir());
  • 输出结果

外部存储

外部存储首先分为私有目录和公共目录,同时是通过 Environment 类来获取使用,其中涉及到WRITE_EXTERNAL_STORAGE的权限申请

/sdcard/xxx
// 返回值:获取外部存储的根目录 /storage/emulated/0
Environment.getExternalStorageDirectory();
// 返回值:/storage/emulated/0/DCIM, 另外还有MOVIE/MUSIC等很多种标准路径
Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM);   

/storage/emulated/0/Android/data/包名/
// 返回值:/storage/emulated/0/Android/data/包名/files
context.getExternalFilesDir();
// 返回值:/storage/emulated/0/Android/data/包名/cache
context.getExternalCacheDir();

demo自测(oppo手机 - 项目地址:手机存储 - Android - data - com.example.storage)

使用方式

Log.e("FileTag", "getExternalFilesDir())-File path:" + getExternalFilesDir(null));
 Log.e("FileTag", "getExternalCacheDir())-File path:" + getExternalCacheDir());
  • 输出结果

补充:获取下载目录 Environment.getDownloadCacheDirectory()

Look here:根据源码文档中说明, 获取外部存储时, 有可能会因为各种问题导致获取失败, 建议先使用 getExternalStorageState 来判断外部存储状态, 如果已挂载的话再存储

通过读取mounts节点获取SD卡挂载状态

private boolean isSdcardMounted(){
  boolean mounted = false;
  String line = null;
  BufferedReader reader = null;
  try{
    reader = new BufferedReader(new FileReader("/proc/mounts"));
    if(reader == null){
      return mounted;
    }
    while ((line = reader.readLine()) != null) {
      String[] tmp = line.split(" ");
      if(tmp.length >= 2){
        if(tmp[1] != null&& tmp[1].equals("/storage/sdcard1")){
          mounted = true;
          break;
        }
      }
    }
  }catch(FileNotFoundException e){
  }catch(IOException ee){
  }finally{
    try{
      if(reader != null)
        reader.close();
    }catch(IOException eee){
    }
  }
  Log.d(TAG,"isSdcardMounted mounted:"+mounted);
  return mounted;
}

在手机设置中的应用设置里有 清除数据 和 清除缓存 两个选项, 它们清除的位置是:

  • Clear Data:清理的是私有目录下的file文件夹

清除数据 清除的是 mnt/sdcard/Android/data/包名/ 和 /data/data/包名/ 下的所有内容;

  • Clear Cache:清理的是私有目录下的cache文件夹

清除缓存 会清除 mnt/sdcard/Android/data/包名/cache/ 和 /data/data/包名/cache/ 内的内容.

end:关于/sdcard/xxx 下的内容依然会存在.~


兴趣扩展

以下均为因个人爱好而进行的二次扩展解惑

android开发在对sd操作时,最好是sd卡处于Environment.MEDIA_MOUNTED 正常状态时,对sd卡上的文件进行操作,其他状态不宜进行操作

如果只是获取当前sd卡状态,不需要对其监听,可以用方法Environment.getExternalStorageState()获得当前sd卡状态

关于SD卡状态监听,到现在为止我知道的有两种方式

1. 注册StorageEventListener来监听sd卡状态

StorageEventListener中有onStorageStateChanged()方法,当sd卡状态改变时,此方法会调用,对各状态的判断一般会用到Environment类,此类中包含的有关sd卡状态的常量有:

状态

含义

Environment.MEDIA_MOUNTED

sd卡在手机上正常使用状态(表明sd对象是存在并具有读/写权限)

Environment.MEDIA_MOUNTED_READ_ONLY

对象权限为只读

Environment.MEDIA_NOFS

对象为空白或正在使用不受支持的文件系统

Environment.MEDIA_UNMOUNTED

用户手动到手机设置中卸载sd卡之后的状态

Environment.MEDIA_REMOVED

用户手动卸载,然后将sd卡从手机取出之后的状态

Environment.MEDIA_BAD_REMOVAL

用户未到手机设置中手动卸载sd卡,直接拨出之后的状态

Environment.MEDIA_SHARED

手机直接连接到电脑作为u盘使用之后的状态(如果 SDCard 未安装 ,并通过 USB 大容量存储共享 返回)

Environment.MEDIA_CHECKINGS

手机正在扫描sd卡过程中的状态(表明对象正在磁盘检查)

Environment.MEDIA_UNMOUNTABLE

返回 SDCard 不可被安装 如果 SDCard 是存在但不可以被安装

Environment.MEDIA_UNMOUNTED

返回 SDCard 已卸掉如果 SDCard 是存在但是没有被安装

如:

public void onStorageStateChanged(String path, String oldState, String newState) {
if (newState.equals(Environment.MEDIA_SHARED)) {
	//大容量存储时相关操作
} else if (newState.equals(Environment.MEDIA_CHECKING)) {
	//检查sd卡时操作
} else if (newState.equals(Environment.MEDIA_MOUNTED)) {
	//sd在手机上可用时操作
	}...
}

StorageEventListener中还有onUsbMassStorageConnectionChanged()用来监听大容量存储是否连接,我对这个方法不太了解,原来以为用来监听usb线是否拔出的,可实际在linux上好像没什么问题,但在windows上却无法用来监听usb拔出状态,期待高手解答。

2. 通过接收Intent来监听sd卡状态

sd卡状态改变时,MountServices会发送Intent,可以通过接收Intent来得到sd卡状态,Intent中关于sd卡状态的action有:

Activity

含义

MEDIA_SCANNER_STARTED_ACTION

开始扫描

MEDIA_SCANNER_FINISHED_ACTION

扫描完成

MEDIA_UNMOUNTED_ACTION

Sd卡存在,但还没有挂载

MEDIA_MOUNTED_ACTION

sd卡被插入,且已经挂载

ACTION_MEDIA_SHARED

Sd卡作为USB大容量存储被共享,挂载被解除

MEDIA_EJECT_ACTION

用户想要移除sd卡

ACTION_MEDIA_REMOVED

sd卡被移除

MEDIA_BAD_REMOVAL_ACTION

sd卡已经从sd卡插槽拔出,但是挂载点还没解除

ACTION_MEDIA_BUTTON

“媒体”按钮被按下

ACTION_MEDIA_CHECKING

存在外部媒体,磁盘检查挂载点的路径中包含的检查媒体意图

ACTION_MEDIA_EJECT

用户移除外部存储媒体

ACTION_MEDIA_NOFS

外部媒体存在,而是使用一个不兼容的fs(或者是空白)挂载点的路径中包含的检查媒体意图

ACTION_MEDIA_SCANNER_SCAN_FILE

请求媒体扫描仪扫描一个文件,并将它添加到媒体数据库


外部应用获取SD卡状态

像是从android4.0以后,外部应用就无法通过android标准接口获取到SD卡状态,但是可以通过如下方法获取:

1.通过env 获取SD卡路径

String externalStorage = System.getenv("SECONDARY_STORAGE");

2. 通过读取mounts节点获取SD卡挂载状态

private boolean isSdcardMounted(){
  boolean mounted = false;
  String line = null;
  BufferedReader reader = null;
  try{
    reader = new BufferedReader(new FileReader("/proc/mounts"));
    if(reader == null){
      return mounted;
    }
    while ((line = reader.readLine()) != null) {
      String[] tmp = line.split(" ");
      if(tmp.length >= 2){
        if(tmp[1] != null&& tmp[1].equals("/storage/sdcard1")){
          mounted = true;
          break;
        }
      }
    }
  }catch(FileNotFoundException e){
  }catch(IOException ee){
  }finally{
    try{
      if(reader != null)
        reader.close();
    }catch(IOException eee){
    }
  }
  Log.d(TAG,"isSdcardMounted mounted:"+mounted);
  return mounted;
}

3. SD卡剩余空间获取

private int getSdcardFreeSpace(){//unit is Million
  int space = 0;
  File file = new File("/storage/sdcard1");
  if(file.exists()){
    long freeSize = file.getUsableSpace();
    space = (int)(freeSize/1024/1024);
  }
  Log.d(TAG,"getSdcardFreeSpace :"+space);
  return space;
}