聊聊对Andorid的FileProvider的理解

news/2024/9/20 11:30:44

最近在项目中遇到一个有关FileProvider的问题,由此来学习了一下FileProvider,有关笔记记录如下。

FileProvider是 Android 系统中一个特殊的内容提供者(ContentProvider), 它主要用于应用之间安全的共享文件。通过 FileProvider,应用可以生成一个 content URI并授予其它应用临时访问的权限,而不需要将文件的实际路径暴露给其它应用。这样可以有效的避免安全问题。

主要用途

FileProvider主要应用于以下几种场景:

  1. 文件共享:当一个应用需要与另外一个应用共享文件时(比如使用Intent发送图片/文件),可以使用 FileProvider来生成一个 content URI;
  2. 文件访问控制:通过 FileProvider 可以控制其他应用对文件的读写权限,确保文件访问的安全性;
  3. 避免使用file://URI:由于从Andorid 7.0起,file://URI被弃用,使用 FileProvider 提供的 content URI 可以避免兼容性。
使用步骤

a. 在 AndroidManifest.xml 中声明 FileProvider:

<providerandroid:name="androidx.core.content.FileProvider"android:authorities="${applicationId}.fileprovider"android:exported="false"android:grantUriPermissions="true"><meta-dataandroid:name="android.support.FILE_PROVIDER_PATHS"android:resource="@xml/file_paths" />
</provider>

b. 配置文件路径

res/xml 目录下创建一个 file_paths.xml 文件,定义共享文件的路径。

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android"><external-files-path name="my_files" path="." />
</paths>

c. 生成 content URI
使用 FileProvider.getUriForFile() 方法来生成一个 content URI。

File file = new File(context.getExternalFilesDir(null), "example.jpg");
Uri contentUri = FileProvider.getUriForFile(context, "com.example.myapp.fileprovider", file);

d. 授予临时权限,并分享

当通过Intent发送文件时,可以授予接收临时访问文件的权限:

Intent intent = new Intent(Intent.ACTION_SEND);
intent.setType("image/*");
intent.putExtra(Intent.EXTRA_STREAM, contentUri);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);startActivity(Intent.createChooser(intent, "Share image via"));

详细解释 file_paths.xml

file_paths.xml, <external-files-path name="my_files" path="." />的解释如下:

  • external-files-path 用于指定应用的外部私有存储目录中的文件路径。外部私有存储目录是指通过 Context.getExternalFilesDir()方法获取的目录,它通常用于存储应用专属的文件,这些文件会随着应用的卸载而被删除。
  • name=“my_files”: 这是一个可选的属性,用于为这个路径配置起一个别名。在生成的Content URI中,这个别名会作为路径的一部分。如果不需要这个别名,那么也可以直接去掉。
  • path="."这个属性指定了相对于 external-files-path 目录的相对路径. "."代表当前目录,即整个 external-files-path 所在的目录。如果你想指定该目录下的某个子目录,你可以在 path属性中提供相对路劲,例如path=“images/”.

我们来一个例子:

假设我们的包名是com.xing.dev,并且在 file_path.xml中配置了如下内容:

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android"><external-files-path name="my_files" path="." />
</paths>

然后通过getExternalFilesDir获取文件保存目录,并且在该目录下有一个a.png文件:

File file = new File(context.getExternalFilesDir(null), "a.png");
Uri contentUri = FileProvider.getUriForFile(context, "com.xing.dev.fileprovider", file);

那么,此时生成的 contentUri就应该是如下的样子:

content://com.xing.dev.fileprovider/my_files/a.png

当然,我们还可以配置更加复杂的路径:

如果我现在只想别的应用仅仅共享特定目录下的文件,那么我就可以配置更加具体的路径:

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">  <external-files-path name="images" path="images/" />  <external-files-path name="docs" path="documents/" />
</paths>

这样,那么就只能访问 /external-files-path /images/ 和 /external-files-path/documents/ 文件夹下的文件了。

那么,除了external-files-path 这个外部私有存储目录中的文件路径,file_paths.xml中还支持哪些tag呢?

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android"><!-- Environment.getExternalStorageDirectory() --><external-path name="external_files" path="." /><!-- Context.getExternalFilesDir(String) --><external-files-path name="external_files_path" path="." /><!-- Context.getExternalCacheDir() --><external-cache-path name="external_cache_path" path="." /><!-- Context.getFilesDir() --><files-path name="internal_files_path" path="." /><!-- Context.getCacheDir() --><cache-path name="internal_cache_path" path="." /></paths>

目前来说,现在支持的是5个,分别是external-path,external-files-path,external-cache-path , files-pathcache-path, 可以通过下表来查看他们之间的含义:

假设我们的包名还是上面那个:com.xing.dev:

标签名称对应java代码路径Demo含义
external-pathEnvironment.getExternalStorageDirectory()/storage/emulated/0整个外部存储目录
external-files-pathContext.getExternalFilesDir(String)/storage/emulated/0/Android/data/com.xing.dev/files应用的外部私有存储目录中的文件路径
external-cache-pathContext.getExternalCacheDir()/storage/emulated/0/Android/data/com.xing.dev/cache应用的外部私有存储目录中的缓存路径
files-pathContext.getFilesDir()/data/user/0/com.xing.dev/files应用的内部私有存储目录中的文件路径
cache-pathContext.getCacheDir()/data/user/0/com.xing.dev/cache应用的内部私有存储目录中的缓存路径

那么,如果我们理解了以上文件目录所代表的含义,那么使用FileProvider就非常容易了。总结下来,我们就可以有以下步骤:

  1. 确定我们需要共享文件的位置,是在整个外部存储目录中(这个显然不是特别安全,不推荐使用),还是在外部/内部私有文件/缓存目录中;
  2. 确定好之后,那么我们可以确定使用哪个标签;
  3. 确定好标签之后,如果还有更细粒度的要求,我们还可以加上子文件夹;
  4. 确定好 file_paths.xml之后,可以通过代码获取对应目标的文件夹,然后生成对应的 ContentURI;
  5. 最后就可以进行分享了。

对于一个越来越安全的android系统,使用FileProvider将App的存储目录分为5个部分,然后通过各自定义文件配置,实现对文件精细粒度的控制。


https://dhexx.cn/news/show-5385018.html

相关文章

QNX 7.0.0开发总结

1 QNX编译 1.1 基本概念 QNX可以直接使用Linux Makefile编译库和二进制&#xff0c;在Makefile文件中指定CCaarch64-unknown-nto-qnx7.0.0-g&#xff0c;或者CCx86_64-pc-nto-qnx7.0.0-g&#xff0c;保存退出后&#xff0c;运行source /qnx_sdk_path/qnxsdp-env.sh&#xff0c;…

iOS18首个Beta测试版发布,功能介绍附beta升级办法!

今天凌晨&#xff0c;一年一度的苹果WWDC24开发者大会正式开幕&#xff0c;发布了iOS 18、iPadOS 18、macOS Sequoia、watch OS11等新系统。 大会结束后&#xff0c;苹果火速发布了首个iOS 18开发者Beta版&#xff0c;目前有开发者资格的用户已经可以下载体验尝鲜了。 本次更新…

C++学习日记 | LAB 7 shared library 共享库

资料来源&#xff1a;南科大 于仕琪 C/C Program Design LINK&#xff1a;CPP/week07 at main ShiqiYu/CPP GitHub 一、本节内容 本节主要介绍建立共享库的内容。习题主要内容为 共享库将开发人员希望与其他开发人员共享的功能的编译代码打包在一起。共享库通常具有以下特点…

从数据采集到部署,手把手带你训练一个高质量的图像分类模型

本文来自社区投稿&#xff0c;作者李剑锋 MMPreTrain 是一款基于 PyTorch 的开源深度学习预训练工具箱&#xff0c;本文将从数据采集到部署&#xff0c;手把手带大家使用 MMPreTrain 算法库训练一个高质量的图像分类模型。 MMPreTrain 项目链接&#xff1a; https://github.co…

python 启动 exe

方法1 import os exe_path "D:\\MyInstaller\\QQ9\\QQ.exe" os.system(exe_path)方法2 import subprocess # 指定exe文件的路径 exe_path "D:\\MyInstaller\\QQ9\\QQ.exe" # 使用subprocess启动exe process subprocess.call(exe_path)方法3 import s…

经典文献阅读之--FlashOcc(快速且内存高效的占用预测模块)

Tip: 如果你在进行深度学习、自动驾驶、模型推理、微调或AI绘画出图等任务&#xff0c;并且需要GPU资源&#xff0c;可以考虑使用UCloud云计算旗下的Compshare的GPU算力云平台。他们提供高性价比的4090 GPU&#xff0c;按时收费每卡2.6元&#xff0c;月卡只需要1.7元每小时&…

抖音快手AI无人直播系统:教你快速搭建视频循环直播场景只需五部

AI无人直播是一种创新的直播方式&#xff0c;利用先进的技术手段实现自动直播&#xff0c;无需人工干预。这种直播方式具有全天候自动直播的能力&#xff0c;无需运营和监管即可吸引流量并转化为订单。商家门店对这种低成本高效果的方式非常欢迎。通过轻松进行直播销售&#xf…

java:使用JSqlParser给sql语句增加tenant_id和deleted条件

# 示例代码 【pom.xml】 <dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-core</artifactId><version>3.4.3.1</version> </dependency>【MyJSqlParserTest.java】 package com.chz.myJSqlParser;pu…

labelme转YOLOv8、YOLOv5 标签格式 标注数据

前言 本文分析将labelme的标签&#xff0c;转为YOLOv8、YOLOv5的格式&#xff0c;实现模型训练。 首先了解YOLOv8和YOLOv5标签格式&#xff0c;然后了解labelme标签格式&#xff0c;最近实现数据格式转换。 1、YOLOv8和YOLOv5标签格式 YOLOv8 的标签格式与 YOLOv5 基本相同&…

MCK主机加固:智能科技,构筑网络安全的铜墙铁壁

在数字化转型的浪潮中&#xff0c;企业服务器的安全已成为维护业务连续性和保护数据资产的关键。MCK主机加固产品&#xff0c;以其创新技术&#xff0c;为企业提供了一个全面、智能、高效的安全解决方案。 一、智能安全监测 MCK主机加固产品采用深度学习算法&#xff0c;能够…