Android文件系统总结——从-Android-Q-“开始”

|

1
2
3
4
5

|

/data/data/{packageName}/files/
context.getFilesDir().getAbsolutePath();

/data/data/{packageName}/cache/
context.getCacheDir().getAbsolutePath();

|

storage 分区(External Storage/Shared Storage)

External Storage/Shared Storage,不需要 root 权限就可以操作。可能包含可移除的存储介质,在使用之前需要判断是否挂载(mounted)

对于 4.4 以前的手机,设备存储很小,存在一个内置的存储空间,这部分空间就是内部存储。另外,还支持一个可以移除的存储介质,就是外部存储,比如 SD 卡。随着硬件升级,大部分大于 Android 4.4 的设备内置的存储空间可以划分为 “内部存储” 和 “外部存储”。同时,若此时还支持插入 SD 卡,则外部存储空间等于 “外部存储” 和 “SD卡存储” 之和。

查看方式

|

1
2
3

|

/storage/emulated/0
@Deprecated  Android10 版本上不再推荐使用该 Api
Environment.getExternalStorageDirectory()

|

主要目录

  • storage/emulated/0/Android/
  • media/{packageName},以包名的形式区分,app的私有多媒体空间,5.0 Api可用
  • obb/{packageName},以包名的形式区分,游戏 obb 数据文件
  • data/{packageName} 以包名的形式区分,app的私有存储空间
  • /cache 缓存空间,可通过 context.getExternalCacheDir() 获取
  • /files 数据存储空间,可通过 context.getExternalFilesDir() 获取, Android 10 通过以下方法进一步操作
  • /Music, 通过 context.getExternalFilesDirs(Environment.DIRECTORY_MUSIC) 获取
  • /Podcasts, 通过 context.getExternalFilesDirs(Environment.DIRECTORY_PODCASTS) 获取
  • /Ringtones, 通过 context.getExternalFilesDirs(Environment.DIRECTORY_RINGTONES) 获取
  • /Alarms, 通过 context.getExternalFilesDirs(Environment.DIRECTORY_ALARMS) 获取
  • /Notifications, 通过 context.getExternalFilesDirs(Environment.DIRECTORY_NOTIFICATIONS) 获取
  • /Pictures, 通过 context.getExternalFilesDirs(Environment.DIRECTORY_PICTURES) 获取
  • /Movies, 通过 context.getExternalFilesDirs(Environment.DIRECTORY_MOVIES) 获取
  • mediaobb,data Android10及以上 按包名为应用划分 沙盒目录,跟随 app 卸载而删除,外部无法访问
  • storage/emulated/0/Music/ Android 10及以上 无法通过路径访问,SAF,MediaStore 可行
  • storage/emulated/0/Pictures/ 访问同上
  • storage/emulated/0/Ringtones/ 访问同上
  • storage/emulated/0/Alarms/ 访问同上
  • storage/emulated/0/Notifications/ 访问同上
  • storage/emulated/0/Podcasts/ 访问同上
  • storage/emulated/0/Movies/ 访问同上
  • storage/emulated/0/Download/ 访问同上
  • storage/emulated/0/DCIM/ 访问同上
  • storage/emulated/0/Documents/ 访问同上
  • storage/emulated/0/Screenshots/ 访问同上
  • storage/emulated/0/Audiobooks/ 访问同上

Android 逻辑角度看

  • App-specific storage

  • 存储类型:应用专用存储,私有目录

  • 使用方法:getFilesDir(),getCacheDir(),getExternalFilesDir(),getExternalCacheDir(),getExternalMediaDirs()

  • 操作权限:内部存储不需要权限,外部存储从 Android4.4 之后也不需要

  • 外部应用访问:无法访问内部存储,Android 10及以后无法外部存储

  • 卸载是否移除:移除

  • Preferences

  • 存储类型:内部私有存储,键值对存在

  • 使用方法:Jetpack Preferences library

  • 操作权限:不需要

  • 外部应用访问:不可以访问

  • 卸载是否移除:移除

  • Databases

  • 存储类型:内部私有存储,持久化结构

  • 使用方法:Room persistence library

  • 操作权限:不需要

  • 外部应用访问:不可以访问

  • 卸载是否移除:移除

  • Shared storage - Media

  • 存储类型:共享存储,比如一些图片,视音频

  • 使用方法:MediaStore API

  • 操作权限:Android9或者更低版本都需要 READ_EXTERNAL_STORAGE 和 WRITE_EXTERNAL_STORAGE。Android10或更高版本在访问外部app才需要

  • 外部应用访问:可以访问,但是需要 READ_EXTERNAL_STORAGE 权限

  • 卸载是否移除:不移除

  • Shared storage - Documents、files

  • 存储类型:共享存储,比如文档,文件

  • 使用方法:Storage Access Framework

  • 操作权限:不需要

  • 外部应用访问:可以访问,文件选择器可以扫描到

  • 卸载是否移除:不移除

一些重要的建议

  1. 存储作用域 scoped-storage
    针对 Android10 及以上设备,可通过 requestLegacyExternalStorage 来控制,默认 false 。低版本设备可以通过设置为 false 来开启这个特性。以下观点均默认 Andriod 10及以上版本默认支持该特性。
  2. 考虑到专用存储空间会随着 app 的卸载而被移除,针对那些用户希望独立于 app 而存在的文件,比如照片,多媒体等,不希望卸载 app 之后被删除则就应该使用共享存储来保留。
  3. 如果希望外部应用可以访问到内部存储的文件,应该使用 FileProvider 结合 FLAG_GRANT_READ_URI_PERMISSION 使用。
  4. getCacheDir() 可能因为设备内部存储空间不足而删除部分缓存文件,确保在读取之前检测文件是否存在
  5. 如果 app 仅在应用内部为用户提供有价值的多媒体信息,则应使用 getExternalFilesDir 来保存这些多媒体信息,例如选择 getExternalFilesDir(Environment.DIRECTORY_PICTURES) 。
  6. 使用 MediaStore 时,系统会自动扫描外部存储并针对媒体进行,目录为 /storage/emulated/0/
  • MediaStore.Images,被存储在 DCIM/ 和 Picture/
  • MediaStore.Video,被存储在 DCIM/Picture/ 和 Movies/
  • MediaStore.Audio,被存储在 Alarms/Audiobooks/Music/Notifications/Podcasts/Ringtones/
  • MediaStore.Downloads,被存储在 Download/, 这类型是 Android10 及以上可用
  • MediaStore.Files,只有 Andriod10及以上可用
  1. 如果尝试使用外部存储的文件原始路径去访问,即使你有 READ_EXTERNAL_STORAGE 权限,在 android 10 及以上也会收到 FileNotFoundException 。应该改成使用 MediaStore API 来访问。
  2. Android 10及以上版本的设备,每一个多媒体文件都有一个应用归属属性,非归属应用访问时需要授权 READ_EXTERNAL_STORAGE 权限。当应用卸载重装之后,访问之前保存的文件也会需要权限,原因是该保存的文件属于之前安装的应用。

Android10及以上实践建议

|

1
2
3
4
5
6
7

|

 

|

  • 读取非共享外部存储已经不需要 READ_EXTERNAL_STORAGE 和 WRITE_EXTERNAL_STORAGE,因为整体是一个 沙箱 涉及,减少权限请求。只有读取其他app的共享外部存储数据才需要,代码向下兼容。

  • 读取共享数据需要请求必要的权限,使用 MediaStore API 进行文件访问

  • 针对其他应用创建的共享文件

  • 访问:请求必要权限 -> ContentResoler 查找并打开文件

  • 写入:当且仅当你的应用是系统某个功能的默认应用才可以,比如照片管理应用,默认音乐应用等。被设置默认角色之后 -> ContentResoler 查找并打开文件,执行编辑、变更操作。

  • 当应用需要进入兼容 scoped-storage ,面向用户的是允许或拒绝应用程序访问整个外部存储的权限,而不是像图片、视频或音乐等单独的共享集合。

  • 应用无法在沙盒目录之外的路径新建文件,包括使用 Environment.getExternalStorageDirectory().

  • 应用无法直接使用路径访问沙盒之外的文件,MediaStore 只能访问公共目录下的多媒体文件`

  • 应用无法分享文件,需要 file:// 类型的 Uri 转换成 content:// 类型

细节编码可直接查看 Android Q 独立存储 一文

参考文档

关于Android文件系统总结的知识就说到这里,喜欢Android开发的同学,可以**点击这里免费获取《Android开发核心知识点笔记》**研读,祝愿大家的技术都越来越成熟。

除此之外,我也分享一些免费的优质资源,包括:Android学习PDF+架构视频+面试文档+源码笔记高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 这几块的内容。分享给大家,非常适合近期有面试和想在技术道路上继续精进的朋友。

或者是想解决面试难题,也可以点击:

《Android学习PDF+架构视频+面试文档+源码笔记》

来获取学习资料

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门**

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 17
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值