在项目开发中经常需要存储一些数据在本地,在存储过程中往往需要设置存储地址,关于存储地址虽然经常用到,但是有些朋友依旧懵懂不知,索性简单记录一下~
Hint:均为自己所想,无法保证是否完全正确
- 存储区别
- 物理角度
- 逻辑角度
- 分区角度
- 存储特点
- 目录特点
- 存储地址(路径)
- 兴趣扩展
so look here
内部存储与外部存储的使用场景,主要看项目需求和自身对项目的考虑,如果有的数据需要长久保存,那么有的开发人员会放在外部存储中;但是我自己的话,更多的还是使用内部存储,防止为手机产生无效垃圾~
在补充内存方面知识的时候,看到了一篇介绍Android文件系统的目录结构
blog,特此借用一下这张图
正式介绍之前呢,我想说明一下常用的内存路径一般有data/data
与 Android/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节点我们可以看做是手机自身的内部存储
~
如看完整篇依旧存在一些疑惑的话,也可以去
这篇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;
}