前言
最近在做下载PDF文件用Intent打开的功能,遇到FileProvider适配的问题。之前做下载升级的时候对这块就迷迷糊糊的,正好这次弄清楚这个
异常
private void open(String path) {
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.addCategory(Intent.CATEGORY_DEFAULT);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(Uri.fromFile(new File(path)), "application/pdf");
startActivity(intent);
}
以上这段代码在7.0以上的系统运行会直接崩溃。
异常信息
Caused by: android.os.FileUriExposedException: file:///storage/emulated/0/%E9%87%91%E6%A1%94%E4%BA%91%E5%88%9B(%E5%90%88%E5%90%8C)/aa.pdf exposed beyond app through Intent.getData()
对于面向 Android 7.0 的应用,Android 框架执行的 StrictMode API 政策禁止在您的应用外部公开 file:// URI。如果一项包含文件 URI 的 intent 离开您的应用,则应用出现故障,并出现 FileUriExposedException 异常。
以上代码中Uri.fromFile(new File(path)
就会生成一个file://...的链接。
解决
官网给出的解决方案:
要在应用间共享文件,您应发送一项 content:// URI,并授予 URI 临时访问权限。进行此授权的最简单方式是使用 FileProvider 类。如需了解有关权限和共享文件的详细信息,请参阅共享文件。
https://developer.android.com/about/versions/nougat/android-7.0-changes.html#accessibility
FileProvider实际上是ContentProvider的一个子类,它的作用也比较明显了,file:///Uri
不能用,那么换个Uri为content://
来替代
1.声明注册provider
在清单文件中加入
FileProvider的包可能是android.support.v4.content.FileProvider ,根据里你导入的appcompat的包
2.编写resource xml file
以上meta-data中的provider_paths需要放到res/xml(如果没有xml文件夹就新建一个)
path即为代表目录下的子目录,比如:
path="pics" />
代表的目录即为:Environment.getExternalStorageDirectory()/pics
。
3.使用FileProvider API
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.addCategory(Intent.CATEGORY_DEFAULT);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
File file = new File(path);
String authority = getApplicationContext().getPackageName() + ".provider";
Uri uri = FileProvider.getUriForFile(this, authority, file);
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.setDataAndType(uri, "application/pdf");
} else {
intent.setDataAndType(Uri.fromFile(new File(path)), "application/pdf");
}
startActivity(intent);
核心代码
File file = new File(path);
String authority = getApplicationContext().getPackageName() + ".provider";
Uri uri = FileProvider.getUriForFile(this, authority, file);
这里的authority和配置在provider必须要一致
这里获取到的uri:content://com.example.test.provider/test/%E9%87%91%E6%A1%94%E4%BA%91%E5%88%9B(%E5%90%88%E5%90%8C)/aa.pdf
以上的格式就是这样的 content://authorities/定义的name属性/文件的相对路径
这里如果清单文件中没有配置或者配置错了android:authorities="${applicationId}.provider"
那么Uri uri = FileProvider.getUriForFile(this, authority, file);
这句会直接报出空指针异常
代码传送门
附上搜集了一些intent
//打开图片文件
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri, "image/*");
//打开PDF文件
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri, "application/pdf");
//打开文本文件
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri, "text/plain");
//打开音频文件
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
intent.putExtra("oneshot", 0);
intent.putExtra("configchange", 0);
intent.setDataAndType(uri, "audio/*");
//打开视频文件
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
intent.putExtra("oneshot", 0);
intent.putExtra("configchange", 0);
intent.setDataAndType(uri, "video/*");
//打开CHM文件
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri, "application/x-chm");
//打开apk文件
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri, "application/vnd.android.package-archive");
//打开PPT文件
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri, "application/vnd.ms-powerpoint");
//打开Excel文件
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri, "application/vnd.ms-excel");
//打开Word文件
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setDataAndType(uri, "application/msword");
资料:
Android 7.0 行为变更 通过FileProvider在应用间共享文件吧
Android使用文件管理器打开指定文件夹,浏览里面的内容