总结系列-Android文件存储相关

总结系列-Android文件存储相关_第1张图片
 

Android开发中有五种数据持久化API:

总结系列-Android文件存储相关_第2张图片
内部存储
 
  • 目录:/data/data/
  • 特点:
    • 每个应用独占一个以包名命名的私有文件夹
    • 在应用卸载时被删除
    • 对MediaScanner不可见
 
内部存储位于系统中很特殊的一个位置,对于设备中每一个安装的 App,系统都会在  data/data/packagename/xxx  自动创建与之对应的文件夹
 
Android可以说是一个Linux操作系统,它的内部存储空间对于应用程序和用户来讲就是“ /data/data “目录。内部存储与外部存储相比有着比较稳定,存储方便,操作简单,更加安全(可以控制访问权限)等优点,而它唯一的缺点就是空间有限。
总结系列-Android文件存储相关_第3张图片
 
对于内部存储路径,我们一般通过以下两种方式获取,内部存储空间的获取都需要使用Context:

context.getFileDir()

对应内部存储的路径为: data/data/packagename/files,但是对于有的手机如:华为,小米等获取到的路径为: data/user/0/packagename/files

context.getCacheDir()

对应内部存储的路径为: data/data/packagename/cache,但是对于有的手机如:华为,小米等获取到的路径为: data/user/0/packagename/cache
总结系列-Android文件存储相关_第4张图片
外部存储
外部存储即storage文件夹或mnt文件夹。需要注意的是storage中有一个sdcard0文件夹,其中又分为公有目录和私有目录:
公有目录:有9大类,比如DCIM、Download等系统为用户创建的文件夹;
私有目录: 即Android文件夹/storage/sdcard/Android/,其中的data文件夹包含了许多包名组成的文件夹。
 
  • 私有目录(private):storage/emulated/0/Android/
    • 特点
      • 每个应用独占以包名命名的私有文件夹
      • 在应用卸载时被删除
      • 对MediaScanner不可见(例外:多媒体文件夹 API 21)
    • 适用场景: 非私密数据,需要随应用卸载删除
  • 公共目录(public):外部存储中除了私有目录外的其他空间
    • 特点
      • 所有应用共享
      • 在应用卸载时不会被删除
      • 对MediaScanner可见
    • 适用场景: 非私密数据,不需要随应用卸载删除
 
总结系列-Android文件存储相关_第5张图片

getExternalCacheDir()

对应外部存储路径:/storage/emulated/0/Android/data/packagename/cache

getExternalFilesDir(String type)

对应外部存储路径:/storage/emulated/0/Android/data/packagename/files
 
其他:
1、Environment. getDataDirectory() = /data
这个方法是获取内部存储的根路径
 
2、 getFilesDir().getAbsolutePath() 
= /data/user/0/packname/files
这个方法是获取某个应用在内部存储中的files路径
 
3、 getCacheDir().getAbsolutePath() 
= /data/user/0/packname/cache
这个方法是获取某个应用在内部存储中的cache路径
 
4、 getDir(“myFile”, MODE_PRIVATE).getAbsolutePath() 
= /data/user/0/packname/app_myFile
这个方法是获取某个应用在内部存储中的自定义路径
 
5、Environment. getExternalStorageDirectory().getAbsolutePath() 
= /storage/emulated/0
这个方法是获取外部存储的根路径
 
6、Environment.getExternalStoragePublicDirectory (Environment.DIRECTORY_DCIM)
 = /storage/emulated/0
这个方法是获取外部存储公有目录
 
7、 getExternalFilesDir(“”).getAbsolutePath() 
= /storage/emulated/0/Android/data/packname/files
这个方法是获取某个应用在外部存储中的files路径
 
8、 getExternalCacheDir().getAbsolutePath() 
= /storage/emulated/0/Android/data/packname/cache
这个方法是获取某个应用在外部存储中的cache路径
 
总结系列-Android文件存储相关_第6张图片
因为外部存储不一定可用,所以返回值可为空或空数组
区分外部存储与SD卡存储
 
File[] files;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
    files = getExternalFilesDirs(Environment.MEDIA_MOUNTED);
    for(File file:files){
        Log.e("main",file);
    }
}
结果:
/storage/emulated/0/Android/data/packname/files/mounted
/storage/B3E4-1711/Android/data/packname/files/mounted
内部存储我们尽量不要去使用。但是当手机没有外部存储时,我们还是得使用内部存储, 检测外部存储是否可用
public static String getFilePath(Context context,String dir) {
    String directoryPath="";
    if (MEDIA_MOUNTED.equals(Environment.getExternalStorageState()) ) {//判断外部存储是否可用
        directoryPath =context.getExternalFilesDir(dir).getAbsolutePath();
        }else{//没外部存储就使用内部存储  
        directoryPath=context.getFilesDir()+File.separator+dir;
        }
        File file = new File(directoryPath);
        if(!file.exists()){//判断文件目录是否存在
        file.mkdirs();
        }
    return directoryPath;
}
 
  外部存储并不总是可用的,因为外部存储可以移除(早期设备)或者作为USB存储设备连接到PC,
访问前 必须检查是否挂载 (mounted):
 
//检测是否可用: 
boolean mExternalStorageAvailable = false;
boolean mExternalStorageWriteable = false;/* 检查外部存储是否可读写 */void updateExternalStorageState() {
    String state = Environment. getExternalStorageState();
    if (Environment.MEDIA_MOUNTED.equals(state)) {
            // 可读写
        mExternalStorageAvailable = mExternalStorageWriteable = true;
    } else if (Environment.MEDIA_MOUNTED_READ_ONLY.equals(state)) {
            // 可读
        mExternalStorageAvailable = true;
        mExternalStorageWriteable = false;
    } else {
        mExternalStorageAvailable = mExternalStorageWriteable = false;
    }}
 
或者:
String externalStorageState = Environment.getExternalStorageState();
if (externalStorageState.equals(Environment.MEDIA_MOUNTED)){
  //sd卡已经安装,可以进行相关文件操作
}
//监听外部存储状态
BroadcastReceiver mExternalStorageReceiver;/* 开始监听 */void startWatchingExternalStorage() {
    mExternalStorageReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
                // 更新状态
            updateExternalStorageState();
        }
    };
    IntentFilter filter = new IntentFilter();
    filter.addAction(Intent.ACTION_MEDIA_MOUNTED);
    filter.addAction(Intent.ACTION_MEDIA_REMOVED);
    // 动态注册广播接收器
    registerReceiver(mExternalStorageReceiver, filter);
    updateExternalStorageState();}/* 停止监听 */void stopWatchingExternalStorage() {
        // 注销广播接收器
    unregisterReceiver(mExternalStorageReceiver);}
 
 
Environment.getExternalStorageState()状态:
MEDIA_UNKNOWN SD卡未知
MEDIA_REMOVED SD卡移除
MEDIA_UNMOUNTED SD卡未安装
MEDIA_CHECKING SD卡检查中,刚装上SD卡时
MEDIA_NOFS SD卡为空白或正在使用不受支持的文件系统
MEDIA_MOUNTED SD卡安装
MEDIA_MOUNTED_READ_ONLY SD卡安装但是只读
MEDIA_SHARED SD卡共享
MEDIA_BAD_REMOVAL SD卡移除错误
MEDIA_UNMOUNTABLE 存在SD卡但是不能挂载,例如发生在介质损坏
 

/storage/sdcard,/sdcard,/mnt/sdcard,/storage/emulated/0之间的关系

在真机上, getExternalStorageDirectory()获取到的路径如下表所示:
系统版本    结果  (注意:下面路径中的文件夹相同)
4.0    /mnt/sdcard
4.1    /storage/sdcard0
4.2    /storage/sdcard0
4.4    /storage/emulated/0
6.0    /storage/emulated/0
其中sdcard/、mnt/sdcard、storage/sdcard0、storage/emulated/0、storage/emulated/legacy都是同一个路径的不同”指针“,指向的是同一个地方,只是不同Android版本的叫法不一样。
 
在Android版本4.2JellyBean之前,获取sdcard的路径是 /sdcard/ ,但在JellyBean版本之后的路径成为了 /sdcard/0 ,或者是 /sdcard/legacy (legacy可以是0、1、2……),这个“0”到底代表着什么含义?
这是从JellyBean版本起的一个新特征——多用户。因此为了处理单独的账户,部分目录结构必须被改变,/sdcard/legacy始终指向当前登录的用户的SD卡目录。
 
正因“多用户”功能的增加,内外部存储发生了以下变化:
内部存储: 原先的/data/data/其实相当于直接链接到当前用户文件夹的,变成了/data/user/0/。
外部存储:例如sd卡路径不再是/sdcard/,而是/sdcard/legacy/(legacy可以是0、1、2……),其中的“0”可以当成“设备拥有者”,或者称为“第一用户”(“第一用户”毫无疑问的是“设备所有者”,只有此用户才能创建额外账户)。
 
 
 
 
 

参考:
Android | 文件存储看这篇就够了
彻底搞懂Android文件存储
Android内、外存储 易混淆点剖析(/mnt/sdcard、/storage/sdcard0、/storage/emulated/0等区别)
一篇文章搞懂android存储目录结构
 
 

你可能感兴趣的:(Android总结系列,[Android开发])