Android 存储进化:分区存储,这些面试官常问的开发面试题你都掌握好了吗
作者:互联网
可移除卷(例如 SD 卡)在文件系统中属于外部存储空间。空间较大,现在的智能机基本都配有,但为了兼容性,也可在使用相关api时检查该空间是否处于可用状态。Environment.getExternalStorageState()
// 是否可读写
fun isExternalStorageWritable(): Boolean {
return Environment.getExternalStorageState() == Environment.MEDIA_MOUNTED
}
// 是否可读
fun isExternalStorageReadable(): Boolean {
return Environment.getExternalStorageState() in
setOf(Environment.MEDIA_MOUNTED, Environment.MEDIA_MOUNTED_READ_ONLY)
}
在写入存储之前,需要知道设备有多少空间可用,不够的话会抛出异常。不过现在的智能设备内存也是比较大的,这部分可以参考Google 查询可用空间
打开Android studio的 Device File Explorer时,可以看到应用的内部空间目录:/data/data/包名/
内部存储空间本身便是保护应用隐私设计的。这部分是没有变化的。应用不需要任何系统权限即可读取和写入这些目录中的文件。其他应用无法访问存储在内部存储空间中的文件。
内部存储空间为应用提供目录。一个目录专为应用的持久性文件而设计,而另一个目录包含应用的缓存文件。内部存储空间是应用专属的,是可以正常使用File
相关api的,所以只要取得路径便可自由发挥:
-
持久性文件根目录File:
context.filesDir()
,/data/data/包名/files/ -
缓存性文件根目录File::
context.cacheDir()
,/data/data/包名/cache/
android也提供了一些简便的api创建删除文件:
context.openFileOutput(filename, Context.MODE_PRIVATE)
、context.openFileInput(filename)
、context.fileList()
、context.getDir(dirName, Context.MODE_PRIVATE)
、context.delefteFile(fileName)
注意:卸载app后,系统会自动移除这些目录释放空间!!
/storage/emulated/0/Android/data/包名
Android 10的分区存储特性,对Android系统的外部存储空间重新设计,外部存储被分为应用私有目录以及共享目录两个部分:
-
应用私有目录:存储应用私有数据,外部存储应用私有目录对应Android/data/包名
-
共享目录:存储其他应用可访问文件, 包含媒体文件、文档文件以及其他文件,对应设备DCIM、Pictures、Alarms, Music, Notifications,Podcasts, Ringtones、Movies、Download等目录。
2.2.1 应用私有空间
与以往相同的是,访问自身的外部存储下的应用私有空间是不需要任何权限的。与内部一样,也有一个目录专为应用的持久性文件而设计,和另一个目录包含应用的缓存文件。也是可以正常使用File
相关api的,所以只要取得路径便可自由发挥。
需要注意的不同点是:开启了分区存储特性后,应用只能访问自身的私有空间,即使获得了存储权限,也无法访问其他应用的私有空间
另外与内部空间的区别是,外部存储空间可能被移除也可能有多个,所以返回的是一个数组,对于返回数组中的第一个元素被视为主外部存储卷。除非该卷已满或不可用,否则请使用该卷。
-
持久性文件:
getExternalFilesDirs(@NonNull Context context, @Nullable String type)
,type根据文件类型可传系统预定义的子目录常量,如图片Environment.DIRECTORY_PICTURES
,此时返回/storage/emulated/0/Android/data/包名/files/Pictures
。或者传null直接返回/storage/emulated/0/Android/data/包名/files
-
缓存性文件:
ContextCompat.getExternalCacheDirs(context)
,/storage/emulated/0/Android/data/包名/cache
注意:卸载app后,系统会自动移除这些目录释放空间!!
=========================================================================
如果用户数据可供或应可供其他应用访问,并且即使在用户卸载应用后也可对其进行保存,请使用共享存储空间。
共享文件类型, 包含媒体文件、文档文件以及其他文件,对应设备DCIM、Pictures、Alarms, Music, Notifications,Podcasts, Ringtones、Movies、Download等目录。Android 分别提供用于获得该类型可共享数据文件Uri的 API:
-
媒体内容:可以使用
MediaStore
API 访问此内容 -
文档和其他文件:系统有一个特殊目录,用于包含其他文件类型,例如 PDF 文档和采用 EPUB 格式的图书。应用可以使用
Storage Access Framework
访问这些文件。
对于共享文件,。以往可以通过data column
获得路径,再使用File API来操作,现在都会返回失败。开启了分区存储特性之后,应用只能通过系统提供的api来向系统请求得到对应文件的Uri
,并通过Uri
生成FileDescriptor
和InputStream
等方式进行文件读写:(简而言之,对于共享文件的增删查改,主要问题在于Uri的获取)
注:android 11 又允许通过路径来访问,系统会自动重定向为Uri。
val resolver = applicationContext.contentResolver
//读
resolver.openFileDescriptor(content-uri, “r”)?.use { pfd ->
val inputStream = FileInputStream(pfd.fileDescriptor)
}
resolver.openInputStream(content-uri).use { stream ->
}
//写
resolver.openFileDescriptor(content-uri, “w”)?.use { pfd ->
val outputStream = FileOutputStream(pfd.fileDescriptor)
}
resolver.openOutputStream(content-uri).use { stream ->
}
//图片bitmap
BitmapFactory.decodeFileDescriptor(pfd.fileDescriptor)
MediaStrore API的增删查改,可参看Google官方指南,主要是通过contentResolver获得对应的uri,这里就不引入了。图片来源
3.1.1 MediaStore 概述
Android系统会自动扫描外部存储空间,将媒体文件按类型添加到系统预定义的Images、Videos、Audio files、Downloaded files集合中。Android Q通过MediaStore.Images、MediaStore.Video、MediaStore.Audio、MediaStore.Downloads 访问相对应共享目录文件资源。预定义集合所对应的目录如下表所示:
| 媒体类型 | Uri | 默认创建目录 | 允许创建目录 |
| — | — | — | — |
| Image | content://media/external/images/media | Pictures | DCIM,Pictures |
| Audio | content://media/external/audio/media | Music | Alarms,Music,Notifications,Podcasts,Ringtones |
| Video | content://media/external/video/media | Movies | DCIM,Movies |
| Download | content://media/external/downloads | Download | Download |
注意:MediaStore.Downloads.EXTERNAL_CONTENT_URI
是Android10版本新增API,用于创建、访问非媒体文件
3.1.1 MediaStore 的变化
-
MediaStore API在共享目录指定目录下创建文件或者访问应用自己创建文件,不需要申请存储权限;
-
MediaStore API访问其他应用在共享目录创建的媒体文件(图片、音频、视频), 需要申请存储权限,未申请存储权限,通过ContentResolver查询不到文件Uri,即使通过其他方式获取到文件Uri,读取或创建文件会抛出异常;
-
MediaStore API不能够访问其他应用创建的非媒体文件(pdf、office、doc、txt等), Android 10 里唯一一种访问其他应用创建的非媒体文件的途径是使用存储访问框架 (Storage Access Framework) 提供的文档选择器。
3.1.2 通过api创建的文件存放到哪里?如何自定义位置?
当通过MediaStore API
创建文件时,文件会默认保存到对应的类型目录,比如图片存到Pictures/
目录下,可以往上查看表格的默认目录及允许目录;
可以使用MediaStore.xxx.Media.RELATIVE_PATH
自己指定要存放的目录或者子目录,如:contentValues.put(MediaStore.Images.Media.RELATIVE_PATH, Environment.DIRECTORY_PICTURES + "/自定义子目录")
,文件就会放在Pictures/自定义子目录/
中;或者使用contentValues.put(MediaStore.Images.Media.RELATIVE_PATH, Environment. DIRECTORY_DCIM)
,将文件放到DCIM/
中
注意:每一种类型都有对应的可允许创建的目录,否则会返回失败。具体可创建目录可以往上查看表格
学习分享
在当下这个信息共享的时代,很多资源都可以在网络上找到,只取决于你愿不愿意找或是找的方法对不对了
很多朋友不是没有资料,大多都是有几十上百个G,但是杂乱无章,不知道怎么看从哪看起,甚至是看后就忘
如果大家觉得自己在网上找的资料非常杂乱、不成体系的话,我也分享一套给大家,比较系统,我平常自己也会经常研读。
2021最新上万页的大厂面试真题
七大模块学习资料:如NDK模块开发、Android框架体系架构…
只有系统,有方向的学习,才能在段时间内迅速提高自己的技术。
这份体系学习笔记,适应人群:
**第一,**学习知识比较碎片化,没有合理的学习路线与进阶方向。
**第二,**开发几年,不知道如何进阶更进一步,比较迷茫。
**第三,**到了合适的年纪,后续不知道该如何发展,转型管理,还是加强技术研究。如果你有需要,我这里恰好有为什么,不来领取!说不定能改变你现在的状态呢!
由于文章内容比较多,篇幅不允许,部分未展示内容以截图方式展示 。如有需要获取完整的资料文档的朋友点击我的【GitHub】免费获取。
说不定能改变你现在的状态呢!
由于文章内容比较多,篇幅不允许,部分未展示内容以截图方式展示 。如有需要获取完整的资料文档的朋友点击我的【GitHub】免费获取。
标签:文件,存储,面试题,面试官,目录,应用,Android,MediaStore 来源: https://blog.csdn.net/wa2231a/article/details/122768852