android内部存储和外部存储

news2024/7/2 20:59:55

我们在开发Android应用的过程中,避免不了要用到数据持久化技术,所谓的数据持久化就是将RAM中的瞬时数据保存到ROM中,保证在App退出或者手机关机后数据不会丢失。我们常用的数据持久化的方式有文件存储,数据库存储,SharedPreference存储等。在window中,当我们存储文件或者数据的时候,我们会选择保存到磁盘的某个目录,而在Android中有两个位置可以让应用进行数据持久化存储—内部存储和外部存储。在开发过程中可以根据不同的场景将数据存储在不同的位置,那究竟什么是内部存储?什么是外部存储?什么时候使用内部存储?什么时候使用外部存储?他们之间区别又是什么?带着这些问题,我们开启今天的探索之旅。

内部存储

一说内部存储,有人可能会和内存混淆在一起,其实这两个概念很好区分,内部存储是用于持久化存储的,属于ROM,手机关机或者退出App数据是不会丢失的,而内存是RAM,退出App或者关机之后数据就会丢失。所以,内部存储不是内存。所谓的内部存储,其实是手机ROM上的一块存储区域,主要用于存储系统以及应用程序的数据。内部存储在Android系统对应的根目录是 /data/data/,这个目录普通用户是无权访问的,用户需要root权限才可以查看。不过我们可以通过Android Studio的View----Tool Windows----Device File Explorer工具来查看该目录,内部存储目录的大致结构如下所示。

内部存储目录

从上图可以看到,/data/data目录是按照应用的包名来组织的,每个应用都是属于自己的内部存储目录,而且目录的名称就是该应用的包名,这个目录是在安装应用的时候自动创建的,当应用被卸载后,该目录也会被系统自动删除。所以,如果你将数据存储于内部存储中,其实就是把数据存储到自己应用包名对应的内部存储目录中。每个应用的内部存储目录都是私有的,也就是说内部存储目录下的文件只能被应用自己访问到,其他应用是没有权限访问的。应用访问自己的内部存储目录时不需要申请任何权限。
一个应用典型的内部存储目录结构如下所示。

 

应用内部存储目录

相信很多人看到内部存储的目录结构,都有似曾相识的感觉,没错我们平常经常和内部存储打交道,只不过我们不知道罢了,下面我们来看下内部存储目录下各个子目录的作用。

  • app_webview:主要用于存储webview加载过程中的数据,例如Cookie,LocalStorage等。
  • cache:主要用于存储使用应用过程中产生的缓存数据。
  • databases:主要用于存储数据库类型的数据。我们平常创建的数据库文件就是存储在这里。
  • files:可以在该目录下存储配置文件,敏感数据等。
  • shared_prefs:用于存储SharedPreference文件。我们使用SharedPreference的时候只指定了文件名,并没有指定存储路径,其实SP的文件就是保存到了这个目录下。

那么有哪些API可以获取到内部存储目录呢,我们主要是使用Context类提供的接口来访问内部存储目录。

1.getDataDir()  //获取的目录是/data/user/0/package_name,即应用内部存储的根目录
2.getFilesDir() //获取的目录是/data/user/0/package_name/files,即应用内部存储的files目录
3.getCacheDir() //获取的目录是/data/user/0/package_name/cache,即应用内部存储的cache目录
4.getDir(String name, int mode) //获取的目录是/data/user/0/package_name/app_name,如果该目录不存在,系统会自动创建该目录。

获取到对应的目录后,我们就可以对目录下的文件进行读写。细心的同学可能会发现代码中获取的内部存储根目录是 /data/user/0,并不是前面提到的/data/data,这是怎么回事呢?因为在Android4.2以后增加了多用户的功能,为了适应多用户的功能,原来的/data/data/相当于直接链接到当前用户文件夹的,变成了/data/user/0/,所以我们代码中打印出来的路径是/data/user/0,而不是/data/data,说白了/data/data和/data/user/0/是一个东西。
内部存储空间容量有限,如果内部存储空间被用完,系统会报内存不足。所以,不要把所有的数据都放到内部存储中。在开发应用过程中,我们可以把较敏感的应用数据放在内部存储中,而其他的数据可以放在外部存储中。那外部存储又是什么呢?下面我们接着来学习外部存储。

外部存储

我们知道内部存储中的数据对应用来说是私密的,用户和其他应用都没有访问权限,而外部存储中的数据是可以被其他应用或用户访问甚至删除的,用户可以通过USB方式和PC之间交互外部存储中的数据。我们平常在Android手机的文件管理工具下看到的目录其实就是外部存储。在Android4.4以前,外部存储就是指SD卡,手机自带的存储就是内部存储;但是在Android4.4以后,随着手机机身存储越来越大,手机的机身存储已经可以满足大多数用户的需求,所以很多手机都不需要再安装SD卡。此时外部存储和内部存储都位于手机机身存储上,他们只是同一个存储介质上的不同存储区域。但是很多手机还是保留了SD卡插槽,方便用户自行拓展。如果手机安装了SD卡,那么很显然SD卡目录也属于外部存储目录。这时手机都有了两个外部存储空间,一个位于手机机身存储上,一个位于SD卡上。但是随着机身存储越累越大,SD卡一般可能只适用于转移文件,对于一般应用来说应该也不会把数据写到外置的SD卡上了,所以这里主要以机身存储为例来分析外部存储。
和内部存储不同的是,外部存储根据存储特点不同分为两种类型:外部私有存储和外部共有存储。先来看外部私有存储。

外部私有存储目录

通常来说,应用涉及到的持久化数据一般分为两类:应用相关数据和应用无关数据。前者是指应用使用的数据信息,比如一些配置信息,调试信息,缓存文件等。当应用被卸载,这些信息也应该被随之删除,避免占用不必要的存储空间。例如下面两种场景。

  • 在用户使用应用过程中,产生的文件,图片,视频,音频等数据,这些数据不太敏感但是占用空间比较大,卸载App时不希望这些数据继续保留在用户手机中。
  • 当应用发生闪退时,希望把一些闪退信息保存下来,让用户获取闪退信息文件后通过特定渠道发送给开发人员进行问题定位。同样的,这些信息在卸载App后也不希望继续留在用户手机中。

对于问题一,我们可以直接把数据存储在内部存储中,但是考虑到内部存储空间有限,把这些数据存储到内部存储会浪费内部存储的空间。对于问题二,普通用户(指没有root权限的用户)无法直接查看其中的文件,把数据直接存储在内部存储中是行不通的。这些数据有一个共同点就是他们的生命周期和应用是一致的,而且不太适合于放在内部存储中。为了存储这种类型的数据,Android规定来一个专门的存储空间,这个空间被称为外部私有存储空间。外部私有存储空间属于外部存储,对于某个应用来说,外部私有存储的根目录(这里暂时不考虑SD卡)是 /storage/emulated/0/Android/data/package_name,这个目录有点类似于内部存储目录,都是以包名来命名私有存储空间的。外部私有存储空间有以下特点

  • 内部私有存储中的数据会随着App的卸载一起删除
  • 仅仅安装应用不会在/storage/emulated/0/Android/data/目录下生成该应用的外部私有存储目录,只有在应用中调用API访问外部私有存储目录时,才会创建以package_name命名的私有存储目录。
  • App在访问自己的外部私有存储目录时不需要任何权限
  • 自 Android 7.0 开始,系统对外部存储目录中 应用私有目录的访问权限进一步限制。其他 App 无法通过 file:// 这种形式的 Uri 直接读写其他应用的外部私有存储目录,而是需要通过 FileProvider 访问。

在代码中我们可以通过以下方式来获取外部私有存储目录。

1.getExternalCacheDir() 
/*获取到的目录是/storage/emulated/0/Android/data/package_name/cache,如果该目录不存在,调用这个方法会自动创建该目录。*/
2.getExternalFilesDir(String type) 
/* 1.如果type为"",那么获取到的目录是 /storage/emulated/0/Android/data/package_name/files
   2.如果type不为空,则会在/storage/emulated/0/Android/data/package_name/files目录下创建一个以传入的type值为名称的目录,例如你将type设为了test,那么就会创建/storage/emulated/0/Android/data/package_name/files/test目录,这个其实有点类似于内部存储getDir方法传入的name参数。但是android官方推荐使用以下的type类型
   public static String DIRECTORY_MUSIC = "Music";
   public static String DIRECTORY_PODCASTS = "Podcasts";
   public static String DIRECTORY_RINGTONES = "Ringtones";
   public static String DIRECTORY_ALARMS = "Alarms";
   public static String DIRECTORY_NOTIFICATIONS = "Notifications";
   public static String DIRECTORY_PICTURES = "Pictures";
   public static String DIRECTORY_MOVIES = "Movies";
   public static String DIRECTORY_DOWNLOADS = "Download";
   public static String DIRECTORY_DCIM = "DCIM";
   public static String DIRECTORY_DOCUMENTS = "Documents";*/
外部共有存储目录

外部存储目录还有一个存储空间就是外部共有存储目录,顾名思义,外部共有存储目录存储的数据无论对应用还是用户都是可见的应用只要有外部访问权限,就可以读取外部公共目录下的文件。外部公共目录主要存放和应用无关的数据,这些数据在卸载App的时候不会被删除。外部共有存储目录有以下特点。

  • 当卸载App时,共有存储目录下的文件不会被删除
  • 应用在访问外部公有目录之前,首先要申请外部存储权限,在Android6.0以后,外部存储权限还要动态申请。
  • 任何应用只要有外部存储权限,都可以访问共有存储目录下的数据。

在代码中,我们可以通过以下方式来访问外部公共存储目录:

1.Environment.getExternalStorageDirectory() 
//获取到的目录是/storage/emulated/0,这个也是外部存储的根目录。
2.Environment.getExternalStoragePublicDirectory(String type) 
/* 1.如果type为"",那么获取到的目录是外部存储的根目录即  /storage/emulated/0
   2.如果type不为空,则会在/storage/emulated/0目录下创建一个以传入的type值为名称的目录,例如你将type设为了test,那么就在外部存储根目录下创建test目录,这个方法和getExternalFilesDir的用法一样。android官方推荐使用以下的type类型,我们在SK卡的根目录下也经常可以看到下面的某些目录。
   public static String DIRECTORY_MUSIC = "Music";
   public static String DIRECTORY_PODCASTS = "Podcasts";
   public static String DIRECTORY_RINGTONES = "Ringtones";
   public static String DIRECTORY_ALARMS = "Alarms";
   public static String DIRECTORY_NOTIFICATIONS = "Notifications";
   public static String DIRECTORY_PICTURES = "Pictures";
   public static String DIRECTORY_MOVIES = "Movies";
   public static String DIRECTORY_DOWNLOADS = "Download";
   public static String DIRECTORY_DCIM = "DCIM";
   public static String DIRECTORY_DOCUMENTS = "Documents";*/

外部存储和内部存储对比

要区分外部存储和内部存储,我们最好从逻辑上来理解这两个概念,而不是从物理上。虽然在Android4.4以前,逻辑上和物理上是统一的,但是Android4.4以后,随着外置SD卡的使用越来越少,内部存储和外部存储和物理介质的内外就没有任何关系了。首先通过一个图来说明下外部存储和内部存储与物理存储的关系。

外部存储、内部存储与物理存储的关系

外部存储和内部存储的对比如下表所示。

外部存储和内部存储对比


明白了内部存储和外部存储的区别,在开发的过程中,我们就可以根据我们的需求来选择对应的存储空间了。
 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

动手实现条件随机场(下)

引言 本文基于PyTorch实现条件随机场,实现CRF层参考论文Neural Architectures for Named Entity Recognition中关于CRF层的描述。包含大量的图解和例子说明,看完一定能理解! 论文地址: https://arxiv.org/pdf/1603.01360.pdf 也可…

chatgpt赋能python:Python搜索算法:如何提高你的搜索体验

Python 搜索算法:如何提高你的搜索体验 在当今信息爆炸的时代,搜索已成为许多人获取信息的主要途径。而 Python 的搜索算法,也在此背景下日渐受到重视。本篇文章将深入探讨 Python 搜索算法,介绍以及如何使用它来提高你的搜索体验…

SpringBoot整合模板引擎Thymeleaf(5)

版权声明 本文原创作者&#xff1a;谷哥的小弟作者博客地址&#xff1a;http://blog.csdn.net/lfdfhl 概述 在本节教程中&#xff0c;我们在之前案例的基础上详细介绍利用Thymeleaf实现国际化。 项目结构 依赖文件 请在pom.xml文件中添加如下依赖。 <?xml version"…

Android 9 蓝牙协议初始化

先讲一下Application类的使用 要使用自定义的Application&#xff0c;首先就是要自己新建一个Application的子类&#xff0c;然后把它的名字写在manifest文件里面的application标签里的android:name属性就行&#xff0c;如我的Application子类名字是BaseApplication&#xff0c…

java入门3(程序流程结构)

目录 大部分和C语言一样 顺序结构 选择结构 简单if语句 ​ 多重if结构 嵌套if结构 Switch选择结构 if和Switch嵌套 循环结构 while循环 do while 语句 for循环 break和continue break 中断指令&#xff0c;结束所在层的循环 continue:中断指令 中断本轮的循环&…

chatgpt赋能python:Python收发短信:简单可靠的解决方案

Python收发短信&#xff1a;简单可靠的解决方案 如果您需要向客户发送定期提醒或通知的短信&#xff0c;则 Python 是一种简单易用的解决方案。在本文中&#xff0c;我们将介绍如何使用 Python 发送和接收短信&#xff0c;并探讨一些流行的短信 API。 什么是短信 API&#xf…

深入分析vfio-user设备实现原理 —— Client侧

文章目录 前言数据结构protocolVFIOUserHdrvfio_user_commandVFIOUserHdr flags 设备模型VFIODeviceVFIODevIOVFIOContIO VFIOPCIDeviceVFIOKernPCIDeviceVFIOUserPCIDevice proxyVFIOUserMsgVFIOProxy 流程详解消息发松流程DMA映射流程 前言 数据结构 protocol VFIO User P…

「解析」YOLOv4模型小结

Paper Yolo v4: https://arxiv.org/abs/2004.10934 Scaled-YOLOv4: Scaling Cross Stage Partial Network Source code:https://github.com/AlexeyAB/darknet Bag of Freebies(BoF) 只增加训练成本&#xff0c;但是能显著提高精度&#xff0c;并不影响推理速度&#xff1b;数据…

Kubernetes使用Istio

Kubernetes使用Istio 1、基本概念 1.1、流量方向 南北流量&#xff08;NORTH-SOURTH-TRAFFIC&#xff09;&#xff1a;客户端到服务器之间通信的流量 东西流量(EAST-WEST-TRAFFIC)&#xff1a;指的是服务器和服务器之间的流量 1.2、Service Mesh 2、安装Istio 2.1、下载 …

【编译、链接、装载九】静态链接

【编译和链接九】静态链接 一、demo二、空间与地址分配1、相似段合并 三、即虚拟地址VMA&#xff08;Virtual Memory Address&#xff09;四、重定位1、add调用2、printf调用——同add2、shared 五、重定位表六、符号解析七、c相关问题1、重复代码消除2、全局构造与析构3、C与A…

从创建到维护:掌握package.json的最佳实践

文章目录 I. 介绍什么是package.jsonpackage.json的作用npm与package.json的关系 II. 创建package.jsonnpm init自动生成package.jsonpackage.json各个字段的含义 III. dependencies和devDependenciesdependencies和devDependencies的区别安装依赖包安装依赖包的版本更新依赖包…

Flink 学习十 FlinkSQL

Flink 学习十 Flink SQL 1. FlinkSQL 基础概念 flink sql 基于flink core ,使用sql 语义方便快捷的进行结构化数据处理的上层库; 类似理解sparksql 和sparkcore , hive和mapreduce 1.1 工作流程 整体架构和工作流程 数据流,绑定元数据 schema ,注册成catalog 中的表 table …

【C语言复习】第七篇、关于C语言关键字的知识

目录 第一部分、常见关键字 1、数据类型关键字 2、流程控制类关键字 第二部分、常用的关键字 1、typedef&#xff08;类型重定义/类型重命名&#xff09; 2、static&#xff08;易混淆const&#xff09; 2.1、static修饰局部变量 2.2、static修饰全局变量 2.3、static修饰…

9.创建provider实例

创建provider网络 controller节点 创建一个provider 网络&#xff0c;网络类型为 external 对于 provider 网络来说&#xff0c;实例通过 2 层&#xff08;桥接网络&#xff09;连接到提供商网 络。 参数说明&#xff1a; --share&#xff1a; 允许所有项目都可以使用该网络…

深度学习-【图像分类】学习笔记8 ShuffleNet

文章目录 8.1 ShuffleNet v1 v2理论讲解ShuffleNet v1ShuffleNet v2 8.2 使用Pytorch搭建ShuffleNet 8.1 ShuffleNet v1 v2理论讲解 ShuffleNet v1 论文链接&#xff1a;https://readpaper.com/paper/2963125010 Channel shuffle 相关链接&#xff1a;深度学习-【图像分类】…

车载以太网 - 传输层 - TCP/IP

目录 一、传输层基础介绍 传输层主要包括两种协议 传输层端口号 二、UDP通信 UDP协议介绍 UDP 通信特点: UDP Segment结构 UDP通信过程 三、TCP通信 TCP通信特点: TCP Segment结构 一、传输层基础介绍 传输层的寻址方式&#xff1a;端口号 包括传输层的寻址方式&…

几个SQL的高级写法

一、ORDER BY FLELD() 自定义排序逻辑 MySql 中的排序 ORDER BY 除了可以用 ASC 和 DESC&#xff0c;还可以通过 ORDER BY FIELD(str,str1,...) 自定义字符串/数字来实现排序。这里用 order_diy 表举例&#xff0c;结构以及表数据展示&#xff1a; ORDER BY FIELD(str,str1,..…

chatgpt赋能python:Python支持跨平台软件开发

Python支持跨平台软件开发 作为一种高级编程语言&#xff0c;Python 以其丰富的库和跨平台支持而备受开发人员欢迎。Python 通过将应用程序的可移植性最大化&#xff0c;使得开发人员可以轻松地在不同的操作系统平台上构建和部署软件。 跨平台支持 Python 支持各种不同的操作…

三子棋都玩过吧,那C语言现造一个呢???

目录 前言 三子棋简介 棋盘介绍 规则介绍 程序设计 基本流程 游戏逻辑 菜单界面打印 创建棋盘并初始化 打印棋盘 玩家落子 电脑落子 判断胜负 1.判定是否和棋 2.判定胜负 代码总汇 ✅Game.h 头文件 ✅Game.c ✅Test.c 前言 &#x1f970;想必各位大佬们上学的…

编译安装以及升级Nginx

目录 一、前言 1、简介 2、 Nginx模块 3、与Apache的差异 4、优点 二、编译安装 1、关闭防火墙 2、安装依赖包 3、创建运行用户与组 4、编译安装 5、检测配置文件是否正确 6、添加系统服务 三、版本升级 四、总结 一、前言 1、简介 Nginx是一个高性能的HTTP和反…