赞
踩
FileProvider
是 ContentProvider
的一个特殊子类,它可以为应用生成关联的 content://
内容 URI ,而不是 file:///
类型的 URI,使得应用能够实现安全地共享文件。
内容 URI 允许授予对文件临时的读/写访问权限。当您构建一个包含内容 URI 的 Intent
,并要将包含内容 URI 的 Intent
传递给客户端应用,可以通过 Intent.setFlags()
API 添加访问权限,这些权限在客户端应用的接收 Activity
处于激活状态时有效(接收 Activity
栈销毁时授权自动失效);如果 Intent
是传递给 Service
,在 Service
运行期间权限有效(Service
停止销毁后授权自动失效)。与之相比,控制 file:///
类型 URI 的访问权限是通过变更文件所在的文件系统的权限来实现的,授予的访问权限是针对所有应用可用的,并且除非您手动修改权限,否则一直有效,因此这类授权是非常不安全的。内容 URI 提供的更高级别的文件访问安全性,让 FileProvider
成为 Android 安全架构基础的关键部分。
FileProvider
由于 FileProvider
的默认功能包含为文件生成内容 URI,因此你不需要在代码中定义 FileProvider
的子类。只需要在 AndroidManifest.xml
清单文件中声明 FileProvider
,在应用清单文件的 <application>
标签内部添加 <provider>
标签来声明 FileProvider
组件。设置 android:name
属性值为 androidx.core.content.FileProvider
(AndriodX);设置 android:authorities
属性为 FileProvider
生成内容 URI 的授权,授权字符串必须保证唯一(通常使用包名组装);设置 android:exported
属性为 false
(FileProvider
不需要对外公开);设置 android:grantUriPermissions
属性值为 true
,允许给文件授予临时访问权限。如下示例代码所示:
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="${applicationId}.owen.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<!-- ...... -->
</provider>
注意事项:
1. 对于您自己的应用,考虑使用应用包名.fileprovider
的方式指定授权(亦可增加其他字符),防止不同应用间出现授权冲突;
2. 如果您需要重写FileProvider
类修改默认实现,在<provide>
标签的android:name
属性值必须为类名全称。
FileProvider
只能为事先指定目录下的文件生成内容 URI。指定目录,也就是在 XML 资源文件中定义存储空间和路径。
首先需要创建一个 XML 资源文件,存放在 res/xml
目录下,XML 文件以 <patchs>
为根节点,在根节点下必须一个或者多个表示存储空间和路径的节点,如下示例所示:
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<files-path name="." path="."/>
<external-path name="." path="."/>
<!-- ...... -->
</paths>
在此 XML 文件中,<path>
可包含以下类型的子节点:
<files-path>
:表示在应用内部存储空间中 files/
子目录,这个目录路径跟 Context.getFilesDir()
返回的一致。<cache-path>
:表示在应用内部存储空间中 cache/
子目录,这个目录路径跟 Context.getCacheDir()
返回的一致。<external-path>
:表示在应用外部存储空间中的根目录,这个目录路径跟 Environment.getExternalStorageDirectory()
返回的一致。<external-files-path>
:表示在应用外部存储空间中 files/
子目录,这个目录路径跟 Context.getExternalFilesDir(String)
、Context.getExternalFilesDir(null)
返回的一致。<external-cache-path>
:表示在应用外部存储空间中 cache/
子目录,这个目录路径跟 Context.getExternalCacheDir()
返回的一致。<external-media-path>
:表示在应用外部存储空间中媒体子目录,这个目录路径跟 Context.getExternalMediaDirs()
返回的一致(注意:这个目录只在 API 21+ 的设备上有效)。在这些表示目录路径的子节点中,都包含以下两个属性:
name
:内容 URI 路径片段。为了增强安全性,这个值用来隐藏文件子目录的详细路径信息,也就是在内容 URI 中,用这个属性值替代子目录的路径信息。path
:需要共享文件所在的子目录详细路径,这个值是真实存在的路径。必须注意的是,这个属性值必须是一个子目录,而不能特定的文件或者一系列文件。你可以通过文件名共享单个文件,但是不能使用通配符指定多个文件。示例:res/xml/file_paths.xml
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<files-path name="picture" path="internal/pic/"/>
<files-path name="database" path="internal/db/"/>
<external-path name="." path="."/>
<external-files-path name="picture" path="picture/"/>
</paths>
讲解:以上的示例中,
<files-path name="database" path="internal/db/"/>
这项声明表示可以共享应用内部存储下files/internal/db
目录极其子目录下的文件。假如一个文件存储在files/internal/db
目录下,在生成的内容 URI 中并不会包含internal/db
片段,而是使用name
属性的 值database
隐藏了真实的路径信息。例如为files/internal/db/init_data.db
生成的内容URI 为content://com.owen.demo.android.owen.fileprovider/database/init_data.db
。
FileProvider
中引用目录配置 在应用清单文件中的 <provider>
标签内部,使用 <meta-data>
子标签引用目录配置 XML 资源,其中 android:name
属性值必须是 android.support.FILE_PROVIDER_PATHS
, android:resources
引用定义好的 XML 资源文件,如下示例所示:
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="${applicationId}.owen.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
使用内容 URI 跟其他应用共享文件,您的应用必须生成内容 URI。配置好 ContentProvider
之后,就可以使用 FileProvider
生成文件的内容 URI。先为文件定义一个 File
实例,然后调用 FileProvider.getUriForFile()
API ,传入<provider>
标签 android:authorities
属性声明的授权以及文件 File
实例,即可生成内容 URI。如下代码所示:
val dbFile = File(appContext.filesDir, "db/init_data.db")
val uri = FileProvider.getUriForFile(appContext, appContext.packageName + ".owen.fileprovider", dbFile);
println(uri.toString())
通过调用 FileProvider.getUriForFile()
API 生成的内容 URI 中包含 <provider>
标签 android:authorities
属性声明的授权,文件目录(XML 中 <meta-data>
指定的目录)相对应的相对路径,是 content://<authorities>/<path>
的形式。以上示例打印出来的内容 URI 如下所示:
content://com.owen.demo.android.owen.fileprovider/database/init_data.db
注意事项:
1.FileProvider.getUriForFile()
只能对 XML 文件声明的目录及其子目录下的文件生成内容 URI;
2.FileProvider.getUriForFile()
的授权(第二个参数)必须和清单文件中provider
定义的授权一致。
生成的文件内容 URI 之后,你可以通过两种方式对内容 URI 授予访问权限,给特定的包名授予访问权限,或者在传递内容 URI 的 Intent
中包含访问权限。
通过调用 Context.grantUriPermission(package, Uri, mode_flags)
API 为 content://
类型的 URI 进行授权,通过第三个参数(mode_flags
)传入 Intent.FLAG_GRANT_READ_URI_PERMISSION
、Intent.FLAG_GRANT_WRITE_URI_PERMISSION
这两个标志之一或者两个都传入。授予的访问权限会一直有效,直到调用 Context.revokeUriPermission(targetPackage, uri, modeFlags)
API 取消授权,或者直到设备重启。
Intent
中授予访问权限将内容 URI 传递给请求方应用,并且赋予对内容 URI 的访问权限,按照以下步骤配置:
Intent
实例对象,通过 Intent.setData()
将内容 URI 添加到 Intent 中;Intent.setFlags()
或者 Intent.addFlags()
接口添加 Intent.FLAG_GRANT_READ_URI_PERMISSION
、Intent.FLAG_GRANT_WRITE_URI_PERMISSION
标志(或者同时添加两个);Activity
的 setResult()
方法。 通过 Intent
授予的内容 URI 访问权限,在接收的 Activity
栈处于激活状态时保持有效,当栈销毁之后,授权将自动失效。授予客户端应用一个 Activity
的访问权限,也将会自动扩展到应用的其他组件。
为了支持在运行 Android 4.1 (API level 16) 和 Android 5.1 (API level 22) (含)的设备,以内容 URI 创建一个
ClipData
对象,并且为ClipData
对象配置访问权限。shareContentIntent.setClipData(ClipData.newRawUri("", contentUri)); shareContentIntent.addFlags( Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
- 1
- 2
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。