在项目开发中经常需要存储一些数据在本地,在存储过程中往往需要设置存储地址,关于存储地址虽然经常用到,但是有些朋友依旧懵懂不知,索性简单记录一下~
Hint:均为自己所想,无法保证是否完全正确
so look here:关于存储方式主要分为内部存储、外部存储俩方面,涉及到三个路径位置
内部存储与外部存储的使用场景,主要看项目需求和自身对项目的考虑,如果有的数据需要长久保存,那么有的开发人员会放在外部存储中;但是我自己的话,更多的还是使用内部存储,防止为手机产生无效垃圾~
在补充内存方面知识的时候,看到了一篇有帮助blog,此处借鉴一下 ~,用来阐述Android文件系统的目录结构
- data/data/apk 的安装目录,需要获取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/节点)
关于内部存储与外部存储的区别,我个人仅找到三个不同的理由来阐释这个问题
物理角度从物理角度看待这个问题(这个区别点,站不稳 - - )
- 内置存储:内部存储
- SD卡:外部存储
通过 Environment.getExternalStorageDirectory()获取到外部存储地址,内部分为私有目录和公共目录
- 内部存储:私有目录
- 外部存储:公共目录
- 内部存储:存储资源会随着app的卸载而自动删除,其他app无法访问当前app内部存储的数据,除了用非法手段或者我们主动暴露
- 外部存储(sdcard):存储资源不会自动删除, 除非用户手动删除, 否则会一直存在, 换句话说就是垃圾… ,不过如果需要其他app使用我们的数据,是有人会这么操作的,不过挺危险的
内部存储
关于内部存储路径有俩种获取方式,主用通过Context 来获取,Google官方建议把数据存储在 /sdcard/Android/data/包名/ 下
/data/data/包名/(需要root权限) // 返回值:/data/data/包名/files context.getFilesDir(); // 返回值:/data/data/包名/cache context.getCacheDir(); /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", "getFilesDir())-File path:" + getFilesDir());
Log.e("FileTag", "getCacheDir())-File path:" + getCacheDir());
Log.e("FileTag", "getExternalFilesDir())-File path:" + getExternalFilesDir(null));
Log.e("FileTag", "getExternalCacheDir())-File path:" + getExternalCacheDir());
- 输出结果
外部存储
外部存储是通过 Environment 类来获取,涉及到WRITE_EXTERNAL_STORAGE的权限申请
/sdcard/xxx // 返回值:获取外部存储的根目录 /storage/emulated/0 Environment.getExternalStorageDirectory(); // 返回值:/storage/emulated/0/DCIM, 另外还有MOVIE/MUSIC等很多种标准路径 Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM);
补充:获取下载目录 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文件夹
清除数据 清除的是 /Android/data/包名/ 和 /data/data/包名/ 下的所有内容;
- Clear Cache:清理的是外部存储中的应用私有目录下的cache文件夹
清除缓存 会清除 /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;
}



