其实很早之前我的应用就已经兼容到Android7.0了,此次写这个文章就是想详细梳理一下android的文件系统,以及做一下FileProvider的解析。
Android7.0 (N) 开始,将严格执行 StrictMode 模式,也就是说,将对安全做更严格的校验。而从 Android N 开始,将不允许在 App 间,使用 file:// 的方式,传递一个 File ,否者会抛出 FileUriExposedException 的错误,会直接引发 Crash。
但是,既然官方对文件的分享做了一个这么强硬的修改(直接抛出异常),实际上也提供了解决方案,那就是 FileProvider,通过 content:// 的模式替换掉 file:// ,同时,需要开发者主动升级 targetSdkVersion 到 24 才会执行此策略。
FileProvider是android support v4包提供的,是ContentProvider的子类,便于将自己app的数据提供给其他app访问。
在app开发过程中需要用到FileProvider的主要有
- 相机拍照以及图片裁剪
- 调用系统应用安装器安装apk(应用升级)
authorities:一个标识,在当前系统内必须是唯一值,一般用包名。
exported:表示该 FileProvider 是否需要公开出去。
granUriPermissions:是否允许授权文件的临时访问权限。这里需要,所以是 true。
2、在res的建xml目录,放入provider_paths.xml文件
/paths>
之后FileProvider的path解析策略如下
private static FileProvider.PathStrategy parsePathStrategy(Context context, String authority) throws IOException, XmlPullParserException { FileProvider.SimplePathStrategy strat = new FileProvider.SimplePathStrategy(authority); ProviderInfo info = context.getPackageManager().resolveContentProvider(authority, 128); XmlResourceParser in = info.loadXmlMetaData(context.getPackageManager(), "android.support.FILE_PROVIDER_PATHS"); if (in == null) { throw new IllegalArgumentException("Missing android.support.FILE_PROVIDER_PATHS meta-data"); } else { int type; while((type = in.next()) != 1) { if (type == 2) { String tag = in.getName(); String name = in.getAttributeValue((String)null, "name"); String path = in.getAttributeValue((String)null, "path"); File target = null; if ("root-path".equals(tag)) { target = DEVICE_ROOT; } else if ("files-path".equals(tag)) { target = context.getFilesDir(); } else if ("cache-path".equals(tag)) { target = context.getCacheDir(); } else if ("external-path".equals(tag)) { target = Environment.getExternalStorageDirectory(); } else { File[] externalMediaDirs; if ("external-files-path".equals(tag)) { externalMediaDirs = ContextCompat.getExternalFilesDirs(context, (String)null); if (externalMediaDirs.length > 0) { target = externalMediaDirs[0]; } } else if ("external-cache-path".equals(tag)) { externalMediaDirs = ContextCompat.getExternalCacheDirs(context); if (externalMediaDirs.length > 0) { target = externalMediaDirs[0]; } } else if (VERSION.SDK_INT >= 21 && "external-media-path".equals(tag)) { externalMediaDirs = context.getExternalMediaDirs(); if (externalMediaDirs.length > 0) { target = externalMediaDirs[0]; } } } if (target != null) { strat.addRoot(name, buildPath(target, path)); } } } return strat; } }
root-path 对应DEVICE_ROOT,也就是 File DEVICE_ROOT = new File("/"),即根目录,一般不需要配置。
files-path对应 content.getFileDir() 获取到的目录。
cache-path对应 content.getCacheDir() 获取到的目录
external-path对应 Environment.getExternalStorageDirectory() 指向的目录。
external-files-path对应 ContextCompat.getExternalFilesDirs() 获取到的目录。
external-cache-path对应 ContextCompat.getExternalCacheDirs() 获取到的目录。
| TAG | Value | Path |
|---|---|---|
| TAG_ROOT_PATH | root-path | / |
| TAG_FILES_PATH | files-path | /data/data/ <包名> /files 包名> |
| TAG_CACHE_PATH | cache-path | /data/data/ <包名> /cache 包名> |
| TAG_EXTERNAL | external-path | /storage/emulate/0 |
| TAG_EXTERNAL_FILES | external-files-path | /storage/emulate/0/Android/data/ <包名> /files 包名> |
| TAG_EXTERNAL_CACHE | external-cache-path | /storage/emulate/0/Android/data/ <包名> /cache 包名> |
![]() |
||
| 3、使用,以安装apk为例 |
Intent intent = new Intent(Intent.ACTION_VIEW); intent.addCategory(Intent.CATEGORY_DEFAULT); Uri uri; File file = new File(saveFolder, updateSaveName); if (Build.VERSION.SDK_INT >= 24) {//android 7.0以上 uri = FileProvider.getUriForFile(activity, BuildConfig.APPLICATION_ID.concat(".provider"), file); } else { uri = Uri.fromFile(file); } String type = "application/vnd.android.package-archive"; intent.setDataAndType(uri, type); intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); if (Build.VERSION.SDK_INT >= 24) { intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); } activity.startActivityForResult(intent, 10);
/storage/emulated/999/tencent/MicroMsg/WeiXin/mmexport93.jpg
你一定觉得很奇怪,正常路径是/storage/emulate/0,怎么会有/storage/emulate/999的路径,查找原因是应用分身导致的。之后会抛
java.lang.IllegalArgumentException: Failed to find configured root that contains /storage/emulated/999/tencent/MicroMsg/WeiXin/mmexport93.jpg
那个时候,我的代码的xml的path里面是没有配置root-path节点的。debug时,fileProvide的mRoots是5个元素

后面我添加了root-path节点之后,mRoots变成了6个

之后就完美实现了将path转成Uri。
部分手机可以插外置sdcard,比如红米手机,之后就导致找不到sdcard的root,这时候也是需要配置root-path。
下面在聊一聊Android的文件系统
外部存储的公共目录
DIRECTORY_MUSIC:音乐类型 /storage/emulate/0/music
DIRECTORY_PICTURES:图片类型
DIRECTORY_MOVIES:电影类型
DIRECTORY_DCIM:照片类型,相机拍摄的照片视频都在这个目录(digital camera in memory) /storage/emulate/0/DCIM
DIRECTORY_DOWNLOADS:下载文件类型 /storage/emulate/0/downloads
DIRECTORY_DOCUMENTS:文档类型
DIRECTORY_RINGTONES:铃声类型
DIRECTORY_ALARMS:闹钟提示音类型
DIRECTORY_NOTIFICATIONS:通知提示音类型
DIRECTORY_PODCASTS:播客音频类型
这些可以通过Environment的getExternalStoragePublicDirectory()来获取
public static File getExternalStoragePublicDirectory(String type);
ContentProvider是跨进程的,可以通过数据库方式向外提供数据,提供数据库读写,那么能否提供一些非持久化的数据,比如运行时的内存数据呢,其实也是可以的。ContentProvider提供了call方法,但此方法比较危险,最好配置好访问权限,避免滥用。

你可以重写此方法提供一些内存数据,一些状态值,供其他应用获取,例如
@Nullable @Override public Bundle call(@NonNull String method, @Nullable String arg, @Nullable Bundle extras) {
if (TextUtils.isEmpty(method)) {
return super.call(method, arg, extras); } try {
// 获取A的状态值 if ("getAState".equals(method)) {
if (TextUtils.isEmpty(arg)) {
return super.call(method, arg, extras); } Bundle bundle = new Bundle(); // 获取状态值 bundle.putString(KEY_A, aState); return bundle; } } catch (Exception e) {
e.printStackTrace(); } return super.call(method, arg, extras); }
ContentResolver contentResolver = mContext.getContentResolver(); ContentProviderClient providerClient = contentResolver.acquireUnstableContentProviderClient(ADataProvider.sUri); String aState= "0"; if (providerClient != null) {
Bundle queryAState = contentResolver.call(ADataProvider.sUri, "getAState", null, null); aState= queryPlanetBack.getString(KEY_A); }
致敬前辈,砥砺前行!
发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/198964.html原文链接:https://javaforall.net
