摘要
Android 使用 VFS (Virtual File System) 虚拟文件系统。VFS提供了供存储设备挂载的节点,同一存储设备经过分区后,不同的分区可以挂载到不同的节点上,如手机的内置存储卡。
关键字
内置存储卡 / 外置SD卡
内部存储 / 外部存储
目录结构
VFS 的目录以 /
为根节点,根节点下面又有不同的节点。物理存储设备就是挂载到这些节点上。
-
/data/data
该节点是用户应用的安装目录,如百度地图的安装路径是 /data/data/com.baidu.com,该目录需要root权限 -
/system
该节点是系统应用的安装目录 -
/storage
该节点是内置存储卡和外置SD卡的挂载点
外置SD卡挂载节点:/storage/sdcard1
内置存储卡挂载节点:/storage/emulated/0
不同的设备挂载节点不同,有的可能在 /mnt下
内部存储和外部存储
内部存储卡/外置SD卡
≠ 内部存储/外部存储
首先明确,内置存储卡/外置SD卡 是在物理层面相对于手机大众用户来说的。
外置SD卡:可手动插拔的SD卡。
内置存储卡:焊接在手机内部不可拆卸的存储卡。
而,内部存 /外部存储 是在文件系统逻辑层面相对于开发者来说的,指具体的路径。
一般针对某个应用而言的,属于该应用的存储路径叫内部存储,反之为外部存储。
内部存储
路径:/data/data/package_name
/data/data/
下都是已安装应用的目录,该目录下包含的文件都是以包名作为文件名的目录,例如/data/data/com.sankuai.meituan
- 内部存储的文件是应用的私有文件,其他应用不能访问。
- 应用访问自己的内部存储不需要权限,访问外部存储需要申请权限。
- 应用卸载后,这些文件也会被移除。
获取内部存储的方式如下:
Context
File getDir(String name, String mode) ;
// /data/data/com.sankuai.meituan
File getFilesDir();
// /data/data/com.sankuai.meituan/files
File getCacheDir()
// /data/data/com.sankuai.meituan/cache
String getApplicationInfo().dataDir;
// /data/data/com.sankuai.meituan
其中,参数 mode
指创建模式,一种 4 种
-
MODE_PRIVATE
,设为私有文件 -
MODE_APPEND
,openFileOutput,文件存在则追加内容 -
MODE_WORLD_READABLE
,API 17 废弃 -
MODE_WORLD_WRITEABLE
,API 17 废弃
注意: Android 7.0 以上android.os.Build.VERSION.SDK_INT>=Build.VERSION_CODES.N
使用3/4 常量时,将会导致SecurityException,这意味着不能通过名称共享私有文件。
尝试共享 file://URI
URI将会导致FileUriExposedException,StrictMode API政策禁止在您的应用外部公开file://URL。如果您的应用需要与其他应用共享私有文件,则可以使用 FileProvider 与 FLAG_GRANT_READ_URI_PERMISSION 配合使用。Android 7.0 行为变更 通过FileProvider在应用间共享文件吧
外部存储
外部存储,可以是 外置SD卡 或 内置存储卡的部分分区。
外部存储,分为 公共目录 和 私有目录
版本 | 存储位置 | 是否需要读写权限 |
---|---|---|
Android 4.4以前 | 外部存储(公共目录和私有目录) | 需要 |
Android 4.4以后 | 外部存储(公共目录) | 需要 |
Android 4.4以后 | 外部存储(私有目录) | 不需要 |
外部存储 - 私有目录
- 属于应用私有,但是这些私有数据可以被其他应用访问和修改(通过私有目录的地址)
- 应用卸载时,此目录及其内容将被删除。
- 系统媒体扫描程序不会读取这些目录的文件,因此不能从MediaStore内容提供程序访问这些文件
获取方式
Context
File getExternalCacheDir()
// /storage/emulated/0/Android/data/com.sankuai.meituan/cache
File getExternalFilesDir(String type)
// /storage/emulated/0/Android/data/com.sankuai.meituan/files
外部存储注意事项
- 使用前先检查文件状态
外部存储的文件,所有的应用和用户都能移除和修改这些权限。
在使用外部存储执行工作之前,应该使用 getExternalStorageState() 检查介质是否可用,介质可能已经装载到设备,处于缺失、只读或其他某种状态。
检查可用性的方法:
String status = Environment.getExternalStorageState();
boolean mounted = status.equals(Environment.MEDIA_MOUNTED)
|| status.equals(Environment.MEDIA_MOUNTED_READ_ONLY);
return mounted;
- 对于支持外插SD卡的设备,外部存储包括两部分:内置存储卡和外置SD卡;
Android 4.3 以下,只能通过Context#getExternalFilesDir(type)
来获取外部存储在内置存储卡分区的私有目录,无法获取外置SD卡。
Android 4.3 开始,可以通过Context#getExternalFilesDirs(type)
获取一个File数组,包含了内置存储卡分区和外置SD的私有目录地址。
可以使用兼容库的静态方法 ContextCompate.getExternalFilesDirs() 兼容 4.3。
总结
- 应用使用内部存储不需要权限,内部存储属于应用的私有存储区域,其他应用不可访问,但应用卸载,内部存储中对应的文件也会删除。
- 外部存储分为公共目录和私有目录,外部存储是可以全局访问的,但需要申请读写权限。Android4.4以后私有目录不需要申请读写权限。
- 如果缓存的数据量较大,请不要保存到内部存储中。
- 如果想保存可共享给其他应用的数据,请保存到外部存储的公共目录中。
- clear data 和 clear cache 两个都是应用的缓存数据,清理的是外部存储中的私有目录下的files/cache,即 /storage/emulated/0/Android/data/com.sankuai.meituan/cache
参考资料
感谢以下文章作者
解析Android内部存储、外部存储的区别