Android分区存储到底是怎么回事

文章目录

    • 一、Android存储结构
    • 二、什么是分区存储?
    • 三、私有目录和公有目录
    • 三、存储权限和分区存储有什么关系?
    • 四、我们应该该怎么做适配?
      • 4.1、利用File进行操作
      • 4.2、使用MediaStore操作数据库

一、Android存储结构

Android存储分为内部存储外部存储

在这里插入图片描述

二、什么是分区存储?

看下面的未使用分区存储时的结构图,App私有目录就是上面说的内部存储,共享存储空间就是上面说的外部存储

在这里插入图片描述

分区存储就是在外部存储中的这些文件夹不能随便放了,必须相应的文件类型存到相应的目录中才可以。比如图片文件只能放到Picture目录或者DCIM目录中,就不能放到Movies或者Music中了,否则就会报错崩溃。

在这里插入图片描述

这里提一句,Download目录可以放任何类型的文件,这个目录没有类型限制。

在这里插入图片描述

Android10以前,外部存储中的所有文件虽然有分类目录,但是不管文件是什么类型都可以随便存放,比如mp3音频文件可以放到Movies目录中或者Picture目录中。

对于Android10,Google第一次添加了分区存储方案,这是作为的一个过渡版本,并且Google在Android10上添加了一个属性让你来选择是否使用分区存储方案 ,就是在Manifest中配置的:android:requestLegacyExternalStorage="true",默认是false,即开启分区存储。

从Android11开始,Google强制使用分区存储,也就是说requestLegacyExternalStorage这个属性不再起作用了。

三、私有目录和公有目录

data目录下的可以理解为就是内部存储中的私有目录。
sdcard目录下的就作为公有目录,没有分区存储时,如果要访问里面的文件就需要权限。

在这里插入图片描述

如果使用了分区存储,要注意在sdcard目录中,也有私有目录和公有目录的概念。
在sdcard中的data目录下,可以看到应用的包名,这就是外部存储中的私有目录,访问这里面的文件不需要权限,只有访问Android目录外的文件才需要权限。并且卸载应用还会被删除。

在这里插入图片描述

三、存储权限和分区存储有什么关系?

存储权限也跟Android版本有关,我们很容易把它和分区存储的概念搞在一起弄的晕头转向,其实并没有什么关系,所以我们讲分区存储不需要考虑要什么什么权限,那是另外一回事。

四、我们应该该怎么做适配?

对于Android10以前(不包含Android10),没有分区存储的概念,并且我们操作文件都是用File对象。
对于Android10,我们有两种选择,即可以使用分区存储,也可以不使用。
对于Android10以后(不包含Android10),强制分区存储了,我们操作文件就需要用MediaStore来操作数据库才行。

4.1、利用File进行操作

这个就不过多进行介绍,随便百度都有很多。

4.2、使用MediaStore操作数据库

  1. 我们操作的数据库文件其实就存在内部存储的私有目录中:/data/data/com.android.providers.media
    我们可以将数据库导出利用数据库工具查看(需要root),里面有个files表,可以看到很多字段,这些字段就对应我们着我们平常代码中所写的:Media.DATAMedia.DISPLAY_NAMEMedia.DURATION等等,我们可以根据保存的文件类型以及自己的需求来选择需要的字段。

在这里插入图片描述

  1. 同时可以看到每种类型的文件目录也都有相应的字段,不管是文件夹目录还是文件在数据库中都会有一条数据相对应。

在这里插入图片描述

  1. 使用分区存储后,我们操作文件都需要注意相应的文件类型。
    例如MediaStore中的三种类型媒体:音频,视频,图片。
    每种类型都分别有三种Uri:内部存储,外部存储,可移动存储(这个不用太关心)。
        MediaStore.ImagesMediaStore.VideoMediaStore.Audio
        MediaStore.Images.Media.INTERNAL_CONTENT_URI//content//media/internal/image/mediaMediaStore.Images.Media.EXTERNAL_CONTENT_URI//content//media/external/image/mediaMediaStore.Images.Media.getContentUri(volumeName)//content//media/<volumeName>/image/mediaMediaStore.Video.Media.INTERNAL_CONTENT_URI//content//media/internal/video/mediaMediaStore.Video.Media.EXTERNAL_CONTENT_URI//content//media/external/video/mediaMediaStore.Video.Media.getContentUri(volumeName)//content//media/<volumeName>/video/mediaMediaStore.Audio.Media.INTERNAL_CONTENT_URI//content//media/internal/audio/mediaMediaStore.Audio.Media.EXTERNAL_CONTENT_URI//content//media/external/audio/mediaMediaStore.Audio.Media.getContentUri(volumeName)//content//media/<volumeName>/audio/media

当然MediaStore也还有其他类型,目前一共有五个。
在这里插入图片描述

  1. 我们操作数据库文件每次用到的就是一个Uri,比如说我要插入一张图片,执行完下面这个方法,就可以直接在相册中查看到你添加的图片了。
    private fun insertImage() {val displayName = "test.jpg"val uri = MediaStore.Images.Media.EXTERNAL_CONTENT_URIval values = ContentValues()//根据文件类型和自己的需求选择字段,比如图片这里就必须要指定MIME_TYPEvalues.put(MediaStore.Images.ImageColumns.DISPLAY_NAME, displayName)values.put(MediaStore.Images.ImageColumns.MIME_TYPE, "image/jpg")values.put(MediaStore.Downloads.RELATIVE_PATH, Environment.DIRECTORY_PICTURES)val imageUri = contentResolver.insert(uri, values)//到这里创建的算是一个文件夹,如果通过下面的代码写入数据后就会变成图片文件if (imageUri != null) {try {val bitmap = BitmapFactory.decodeResource(resources, R.mipmap.ic_launcher)val outputStream = contentResolver.openOutputStream(imageUri)if (outputStream != null) {bitmap.compress(Bitmap.CompressFormat.JPEG, 100, outputStream)outputStream.close()}ToastUtil.showToast(this, "添加图片成功")} catch (e: Exception) {e.printStackTrace()}} else {ToastUtil.showToast(this, "操作失败")}}
  1. 简单的查询操作,比如查询上面添加的那张图片。
    private fun query(): Uri? {val uri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI//查询条件,根据DISPLAY_NAMEval selection = MediaStore.Images.Media.DISPLAY_NAME + "=?"val args: Array<String> = arrayOf("test.jpg")val projection: Array<String> = arrayOf(MediaStore.Images.Media._ID)//数据库查询val cursor = contentResolver.query(uri, projection, selection, args, null)return if (cursor != null && cursor.moveToFirst()) {val queryUri = ContentUris.withAppendedId(uri, cursor.getLong(0))cursor.close()ToastUtil.showToast(this, "查询成功:$queryUri")queryUri} else {ToastUtil.showToast(this, "查询失败")null}}
  1. 简单的删除操作,注意删除操作前需要先通过查询得到相应的uri。
    private fun delete() {//先查询后删除val uri = query()if (uri != null) {contentResolver.delete(uri, null, null)}}
  1. 简单的修改操作,跟删除一样,也需要先通过查询得到相应的uri。
    private fun update() {//先查询后修改val uri = query()if (uri != null) {//修改的字段val contentResolver = ContentValues()contentResolver.put(MediaStore.Images.ImageColumns.DISPLAY_NAME, "修改后的图片.jpg")//操作数据库getContentResolver().update(uri, contentResolver, null, null)}}
  1. 总结一下操作数据库,其实就3步:拿到uri,构建字段条件,执行。拿插入图片来举例。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/543957.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

系统重构后,对项目定制开发的兼容性问题

公司自实施产品线战略以来&#xff0c;基本推翻了全部旧有业务模块。后续以标准产品二次开发的模式进行项目开发。但在涉及到一些旧有系统二期、三期升级改造过程中。不可避免的需要解决旧有系统的客户定制化开发兼容性问题。也就是旧有系统定制开发的模块不能丢弃。重新开发从…

HTTPS证书很贵吗?

首先&#xff0c;我们需要明确一点&#xff0c;HTTPS证书的价格并不是一成不变的&#xff0c;它受到多种因素的影响。其中最主要的因素包括证书的类型、颁发机构以及所需的验证级别。 从类型上来看&#xff0c;HTTPS证书主要分为单域名证书、多域名证书和通配符证书。单域名证书…

更安全的C gets()和str* 以及fgets和strcspn的用法

#include <stdio.h>int main() {char *str;gets(str);puts(str);return(0); }可以说全是错误 首先char *str没有指向一个分配好的地址&#xff0c;就直接读入&#xff0c;危险 ps: 怎么理解char *str "Hello World" 是将一个存储在一个只读的数据段中字符串常…

(六)Android布局类型(表格布局TableLayout)

表格布局&#xff08;TableLayout&#xff09;&#xff0c;呈现行列方式&#xff0c;无法设置列&#xff0c;可以设置行&#xff0c;行数由TableRow对象个数决定。下图中有两个TableRow元素&#xff0c;所以&#xff0c;说明表格布局中有两行。 将内容填充到行中 第一行中&…

NFTScan 正式上线 Blast NFTScan 浏览器和 NFT API 数据服务

2024 年 3 月 15 号&#xff0c;NFTScan 团队正式对外发布了 Blast NFTScan 浏览器&#xff0c;将为 Blast 生态的 NFT 开发者和用户提供简洁高效的 NFT 数据搜索查询服务。NFTScan 作为全球领先的 NFT 数据基础设施服务商&#xff0c;Blast 是继 Bitcoin、Ethereum、BNBChain、…

EtherCAT 开源主站 IGH 在 linux 开发板的移植和伺服通信测试

手边有一套正点原子linux开发板imax6ul&#xff0c;一直在吃灰&#xff0c;周末业余时间无聊&#xff0c;把EtherCAT的开源IGH主站移植到开发板上玩玩儿&#xff0c;搞点事情做。顺便学习研究下EtherCAT总线协议及其对伺服驱动器的运动控制过程。实验很有意思&#xff0c;这里总…

vite打包时发布时,放在服务器的二级目录中

方式一 hash模式 如果我们的站点根目录为 public , 我们访问的时候使用的是 http://www.abc.com/ 访问到了站点的根目当&#xff0c;现在我们要访问 http://www.abc.com/mysite/#/ 配置如下 修改 vite.config.js base:“/mysite/” 修改 router中的配置 上面的步骤完成&…

RocketMQ学习笔记四(黑马)项目

课程地址&#xff1a; 1.Rocket第二章内容介绍_哔哩哔哩_bilibili &#xff08;视频35~88&#xff0c;搭建了一个电商项目&#xff09; 待学&#xff0c;待完善。

Microsoft Visio 编辑属性值

Microsoft Visio 编辑属性值 1. 编辑属性值References 1. 编辑属性值 单击长度或高度位置&#xff0c;弹出形状的各属性值&#xff0c;点击编辑对应的属性值。 ​ References [1] Yongqiang Cheng, https://yongqiang.blog.csdn.net/

创业板指399006行情数据API接口

# 测试&#xff1a;返回不超过10条数据&#xff08;2年历史&#xff09; https://tsanghi.com/api/fin/index/CHN/daily?tokendemo&ticker399006&order2Python示例 import requestsurl f"https://tsanghi.com/api/fin/index/CHN/daily?tokendemo&ticker399…

win10 + cpu + pycharm + mindspore

MindSpore是华为公司自研的最佳匹配昇腾AI处理器算力的全场景深度学习框架。 1、打开官网&#xff1a; MindSpore官网 2、选择以下选项&#xff1a; 3、创建conda 环境&#xff0c;这里python 选择3.9.0&#xff0c;也可以选择其他版本&#xff1a; conda create -c conda-…

flink1.18.0 自定义函数 接收row类型的参数

比如sql中某字段类型 array<row<f1 string,f2 string,f3 string,f4 bigint>> 现在需要编写 tableFunction 需要接受的参数如上 解决方案 用户定义函数|阿帕奇弗林克 --- User-defined Functions | Apache Flink